mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 22:26:37 +00:00 
			
		
		
		
	Move mouse related functions and states out of WorldspaceWidget.
This commit is contained in:
		
							parent
							
								
									998982b16a
								
							
						
					
					
						commit
						8b4651f055
					
				
					 6 changed files with 763 additions and 607 deletions
				
			
		|  | @ -82,7 +82,7 @@ opencs_units (view/render | ||||||
| 
 | 
 | ||||||
| opencs_units_noqt (view/render | opencs_units_noqt (view/render | ||||||
|     navigation navigation1st navigationfree navigationorbit lighting lightingday lightingnight |     navigation navigation1st navigationfree navigationorbit lighting lightingday lightingnight | ||||||
|     lightingbright object cell terrainstorage textoverlay overlaymask overlaysystem |     lightingbright object cell terrainstorage textoverlay overlaymask overlaysystem mousestate | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
| opencs_hdrs_noqt (view/render | opencs_hdrs_noqt (view/render | ||||||
|  |  | ||||||
|  | @ -161,6 +161,22 @@ void CSMSettings::UserSettings::buildSettingModelDefaults() | ||||||
|         ritd->setDeclaredValues (values); |         ritd->setDeclaredValues (values); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     declareSection ("debug", "Debug Options"); | ||||||
|  |     { | ||||||
|  |         Setting *mousePicking = createSetting (Type_CheckBox, "mouse-picking", "Debug Render Mouse-Picking"); | ||||||
|  |         mousePicking->setDefaultValue ("false"); | ||||||
|  |         mousePicking->setToolTip ("Enable redering debug information for mouse picking. " | ||||||
|  |             "This option may be removed in future once the mouse picking feature is completed."); | ||||||
|  | 
 | ||||||
|  |         QString defaultValue = "Closer/Further"; | ||||||
|  |         QStringList values = QStringList() << defaultValue << "Up/Down" << "Left/Right"; | ||||||
|  | 
 | ||||||
|  |         Setting *mouseWheel = createSetting (Type_RadioButton, "mouse-wheel", | ||||||
|  |             "For testing mouse movement directions."); | ||||||
|  |         mouseWheel->setDefaultValue (defaultValue); | ||||||
|  |         mouseWheel->setDeclaredValues (values); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     declareSection ("table-input", "Table Input"); |     declareSection ("table-input", "Table Input"); | ||||||
|     { |     { | ||||||
|         QString inPlaceEdit ("Edit in Place"); |         QString inPlaceEdit ("Edit in Place"); | ||||||
|  |  | ||||||
							
								
								
									
										648
									
								
								apps/opencs/view/render/mousestate.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										648
									
								
								apps/opencs/view/render/mousestate.cpp
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,648 @@ | ||||||
|  | #include "mousestate.hpp" | ||||||
|  | 
 | ||||||
|  | #include <OgreSceneNode.h> | ||||||
|  | #include <OgreSceneManager.h> | ||||||
|  | #include <OgreEntity.h> | ||||||
|  | 
 | ||||||
|  | #include <OgreMeshManager.h> | ||||||
|  | #include <OgreManualObject.h>        // FIXME: for debugging
 | ||||||
|  | #include <OgreMaterialManager.h>     // FIXME: for debugging
 | ||||||
|  | #include <OgreHardwarePixelBuffer.h> // FIXME: for debugging
 | ||||||
|  | 
 | ||||||
|  | #include <QMouseEvent> | ||||||
|  | #include <QElapsedTimer> | ||||||
|  | 
 | ||||||
|  | #include "../../model/settings/usersettings.hpp" | ||||||
|  | #include "../world/physicssystem.hpp" | ||||||
|  | 
 | ||||||
|  | #include "elements.hpp" // FIXME: for debugging
 | ||||||
|  | #include "worldspacewidget.hpp" | ||||||
|  | 
 | ||||||
|  | namespace | ||||||
|  | { | ||||||
|  |     // FIXME: this section should be removed once the debugging is completed
 | ||||||
|  |     void showHitPoint(Ogre::SceneManager *sceneMgr, std::string name, Ogre::Vector3 point) | ||||||
|  |     { | ||||||
|  |         if(sceneMgr->hasManualObject("manual" + name)) | ||||||
|  |             sceneMgr->destroyManualObject("manual" + name); | ||||||
|  |         Ogre::ManualObject* manual = sceneMgr->createManualObject("manual" + name); | ||||||
|  |         manual->begin("BaseWhite", Ogre::RenderOperation::OT_LINE_LIST); | ||||||
|  |         manual-> position(point.x,     point.y,     point.z-100); | ||||||
|  |         manual-> position(point.x,     point.y,     point.z+100); | ||||||
|  |         manual-> position(point.x,     point.y-100, point.z); | ||||||
|  |         manual-> position(point.x,     point.y+100, point.z); | ||||||
|  |         manual-> position(point.x-100, point.y,     point.z); | ||||||
|  |         manual-> position(point.x+100, point.y,     point.z); | ||||||
|  |         manual->end(); | ||||||
|  |         sceneMgr->getRootSceneNode()->createChildSceneNode()->attachObject(manual); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void removeHitPoint(Ogre::SceneManager *sceneMgr, std::string name) | ||||||
|  |     { | ||||||
|  |         if(sceneMgr->hasManualObject("manual" + name)) | ||||||
|  |             sceneMgr->destroyManualObject("manual" + name); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void initDebug() | ||||||
|  |     { | ||||||
|  |         // material for visual cue on selected objects
 | ||||||
|  |         Ogre::TexturePtr texture = Ogre::TextureManager::getSingleton().getByName("DynamicTrans"); | ||||||
|  |         if(texture.isNull()) | ||||||
|  |         { | ||||||
|  |             texture = Ogre::TextureManager::getSingleton().createManual( | ||||||
|  |                 "DynamicTrans", // name
 | ||||||
|  |                 Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, | ||||||
|  |                 Ogre::TEX_TYPE_2D,  // type
 | ||||||
|  |                 8, 8,               // width & height
 | ||||||
|  |                 0,                  // number of mipmaps
 | ||||||
|  |                 Ogre::PF_BYTE_BGRA, // pixel format
 | ||||||
|  |                 Ogre::TU_DEFAULT);  // usage; should be TU_DYNAMIC_WRITE_ONLY_DISCARDABLE for
 | ||||||
|  |                                     // textures updated very often (e.g. each frame)
 | ||||||
|  | 
 | ||||||
|  |             Ogre::HardwarePixelBufferSharedPtr pixelBuffer = texture->getBuffer(); | ||||||
|  |             pixelBuffer->lock(Ogre::HardwareBuffer::HBL_NORMAL); | ||||||
|  |             const Ogre::PixelBox& pixelBox = pixelBuffer->getCurrentLock(); | ||||||
|  | 
 | ||||||
|  |             uint8_t* pDest = static_cast<uint8_t*>(pixelBox.data); | ||||||
|  | 
 | ||||||
|  |             // Fill in some pixel data. This will give a semi-transparent colour,
 | ||||||
|  |             // but this is of course dependent on the chosen pixel format.
 | ||||||
|  |             for (size_t j = 0; j < 8; j++) | ||||||
|  |             { | ||||||
|  |                 for(size_t i = 0; i < 8; i++) | ||||||
|  |                 { | ||||||
|  |                     *pDest++ = 255; // B
 | ||||||
|  |                     *pDest++ = 255; // G
 | ||||||
|  |                     *pDest++ = 127; // R
 | ||||||
|  |                     *pDest++ =  63; // A
 | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 pDest += pixelBox.getRowSkip() * Ogre::PixelUtil::getNumElemBytes(pixelBox.format); | ||||||
|  |             } | ||||||
|  |             pixelBuffer->unlock(); | ||||||
|  |         } | ||||||
|  |         Ogre::MaterialPtr material = Ogre::MaterialManager::getSingleton().getByName( | ||||||
|  |                     "TransMaterial"); | ||||||
|  |         if(material.isNull()) | ||||||
|  |         { | ||||||
|  |             Ogre::MaterialPtr material = Ogre::MaterialManager::getSingleton().create( | ||||||
|  |                         "TransMaterial", | ||||||
|  |                         Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, true ); | ||||||
|  |             Ogre::Pass *pass = material->getTechnique( 0 )->getPass( 0 ); | ||||||
|  |             pass->setLightingEnabled( false ); | ||||||
|  |             pass->setDepthWriteEnabled( false ); | ||||||
|  |             pass->setSceneBlending( Ogre::SBT_TRANSPARENT_ALPHA ); | ||||||
|  | 
 | ||||||
|  |             Ogre::TextureUnitState *tex = pass->createTextureUnitState("CustomState", 0); | ||||||
|  |             tex->setTextureName("DynamicTrans"); | ||||||
|  |             tex->setTextureFiltering( Ogre::TFO_ANISOTROPIC ); | ||||||
|  |             material->load(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     //plane Z, upvector Y, mOffset z : x-y plane, wheel up/down
 | ||||||
|  |     //plane Y, upvector X, mOffset y : y-z plane, wheel left/right
 | ||||||
|  |     //plane X, upvector Y, mOffset x : x-z plane, wheel closer/further
 | ||||||
|  |     std::pair<Ogre::Vector3, Ogre::Vector3> planeAxis() | ||||||
|  |     { | ||||||
|  |         CSMSettings::UserSettings &userSettings = CSMSettings::UserSettings::instance(); | ||||||
|  |         QString wheelDir =  userSettings.setting("debug/mouse-wheel", QString("Closer/Further")); | ||||||
|  |         if(wheelDir == "Up/Down") | ||||||
|  |             return std::make_pair(Ogre::Vector3::UNIT_Z, Ogre::Vector3::UNIT_Y); | ||||||
|  |         else if(wheelDir == "Left/Right") | ||||||
|  |             return std::make_pair(Ogre::Vector3::UNIT_Y, Ogre::Vector3::UNIT_X); | ||||||
|  |         else | ||||||
|  |             return std::make_pair(Ogre::Vector3::UNIT_X, Ogre::Vector3::UNIT_Y); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | namespace CSVRender | ||||||
|  | { | ||||||
|  |     // mouse picking
 | ||||||
|  |     // FIXME: need to virtualise mouse buttons
 | ||||||
|  |     //
 | ||||||
|  |     // State machine:
 | ||||||
|  |     //
 | ||||||
|  |     // [default] mousePressEvent->check if the mouse is pointing at an object
 | ||||||
|  |     //         if yes, create collision planes then go to [grab]
 | ||||||
|  |     //         else check for terrain
 | ||||||
|  |     //
 | ||||||
|  |     // [grab]  mouseReleaseEvent->if same button and new obj, go to [edit]
 | ||||||
|  |     //         mouseMoveEvent->if same button, go to [drag]
 | ||||||
|  |     //         other mouse events or buttons, go back to [default] (i.e. like 'cancel')
 | ||||||
|  |     //
 | ||||||
|  |     // [drag]  mouseReleaseEvent->if same button, place the object at the new
 | ||||||
|  |     //         location, update the document then go to [edit]
 | ||||||
|  |     //         mouseMoveEvent->update position to the user based on ray to the collision
 | ||||||
|  |     //         planes and render the object at the new location, but do not update
 | ||||||
|  |     //         the document yet
 | ||||||
|  |     //
 | ||||||
|  |     // [edit]  TODO, probably fine positional adjustments or rotations; clone/delete?
 | ||||||
|  |     //
 | ||||||
|  |     //
 | ||||||
|  |     //               press               press (obj)
 | ||||||
|  |     //   [default] --------> [grab] <-------------------- [edit]
 | ||||||
|  |     //       ^       (obj)    |  |  ------> [drag] ----->   ^
 | ||||||
|  |     //       |                |  |   move     ^ |  release  |
 | ||||||
|  |     //       |                |  |            | |           |
 | ||||||
|  |     //       |                |  |            +-+           |
 | ||||||
|  |     //       |                |  |            move          |
 | ||||||
|  |     //       +----------------+  +--------------------------+
 | ||||||
|  |     //            release                  release
 | ||||||
|  |     //           (same obj)               (new obj)
 | ||||||
|  |     //
 | ||||||
|  |     //
 | ||||||
|  | 
 | ||||||
|  |     MouseState::MouseState(WorldspaceWidget *parent) | ||||||
|  |         : mParent(parent), mPhysics(parent->getPhysics()), mSceneManager(parent->getSceneManager()) | ||||||
|  |         , mCurrentObj(""), mMouseState(Mouse_Default), mOldPos(0,0), mMouseEventTimer(0), mPlane(0) | ||||||
|  |         , mGrabbedSceneNode(""), mOrigObjPos(Ogre::Vector3()), mOrigMousePos(Ogre::Vector3()) | ||||||
|  |         , mCurrentMousePos(Ogre::Vector3()), mOffset(0.0f) | ||||||
|  |     { | ||||||
|  |         initDebug(); | ||||||
|  | 
 | ||||||
|  |         mMouseEventTimer = new QElapsedTimer(); | ||||||
|  |         mMouseEventTimer->invalidate(); | ||||||
|  | 
 | ||||||
|  |         std::pair<Ogre::Vector3, Ogre::Vector3> planeRes = planeAxis(); | ||||||
|  |         mPlane = new Ogre::Plane(planeRes.first, 0); | ||||||
|  |         Ogre::MeshPtr mesh = Ogre::MeshManager::getSingleton().createPlane("mouse", | ||||||
|  |             Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, | ||||||
|  |             *mPlane, | ||||||
|  |             300000,300000, // FIXME: use far clip dist?
 | ||||||
|  |             1,1, // segments
 | ||||||
|  |             true,  // normals
 | ||||||
|  |             1,     // numTexCoordSets
 | ||||||
|  |             1,1, // uTile, vTile
 | ||||||
|  |             planeRes.second // upVector
 | ||||||
|  |             ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     MouseState::~MouseState () | ||||||
|  |     { | ||||||
|  |         delete mMouseEventTimer; | ||||||
|  | 
 | ||||||
|  |         // For debugging only
 | ||||||
|  |         std::map<std::string, std::vector<std::string> >::iterator iter = mSelectedEntities.begin(); | ||||||
|  |         for(;iter != mSelectedEntities.end(); ++iter) | ||||||
|  |         { | ||||||
|  |             removeHitPoint(mSceneManager, iter->first); | ||||||
|  | 
 | ||||||
|  |             if(mSceneManager->hasSceneNode(iter->first)) | ||||||
|  |             { | ||||||
|  |                 Ogre::SceneNode *scene = mSceneManager->getSceneNode(iter->first); | ||||||
|  | 
 | ||||||
|  |                 if(scene) | ||||||
|  |                 { | ||||||
|  |                     scene->removeAndDestroyAllChildren(); | ||||||
|  |                     mSceneManager->destroySceneNode(iter->first); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         delete mPlane; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::mouseMoveEvent (QMouseEvent *event) | ||||||
|  |     { | ||||||
|  |         switch(mMouseState) | ||||||
|  |         { | ||||||
|  |             case Mouse_Grab: | ||||||
|  |             { | ||||||
|  |                 // check if min elapsed time to stop false detection of drag
 | ||||||
|  |                 if(!mMouseEventTimer->isValid() || !mMouseEventTimer->hasExpired(100)) // ms
 | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 mMouseEventTimer->invalidate(); | ||||||
|  |                 mMouseState = Mouse_Drag; | ||||||
|  | 
 | ||||||
|  |                 /* FALL_THROUGH */ | ||||||
|  |             } | ||||||
|  |             case Mouse_Drag: | ||||||
|  |             { | ||||||
|  |                 if(event->pos() != mOldPos) // TODO: maybe don't update less than a quantum?
 | ||||||
|  |                 { | ||||||
|  |                     mOldPos = event->pos(); | ||||||
|  | 
 | ||||||
|  |                     // ray test against the plane to provide feedback to the user the
 | ||||||
|  |                     // relative movement of the object on the x-y plane
 | ||||||
|  |                     std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); | ||||||
|  |                     if(planeResult.first) | ||||||
|  |                     { | ||||||
|  |                         if(mGrabbedSceneNode != "") | ||||||
|  |                         { | ||||||
|  |                             std::pair<Ogre::Vector3, Ogre::Vector3> planeRes = planeAxis(); | ||||||
|  |                             Ogre::Vector3 pos = mOrigObjPos + planeRes.first*mOffset; | ||||||
|  |                             //pos.z += mOffset;
 | ||||||
|  |                             mSceneManager->getSceneNode(mGrabbedSceneNode)->setPosition(pos+planeResult.second-mOrigMousePos); | ||||||
|  |                             mCurrentMousePos = planeResult.second; | ||||||
|  |                             mPhysics->moveSceneNodes(mGrabbedSceneNode, pos+planeResult.second-mOrigMousePos); | ||||||
|  |                             updateSceneWidgets(); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             case Mouse_Edit: | ||||||
|  |             case Mouse_Default: | ||||||
|  |             { | ||||||
|  |                 break; // error event, ignore
 | ||||||
|  |             } | ||||||
|  |             /* NO_DEFAULT_CASE */ | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::mousePressEvent (QMouseEvent *event) | ||||||
|  |     { | ||||||
|  |         switch(mMouseState) | ||||||
|  |         { | ||||||
|  |             case Mouse_Grab: | ||||||
|  |             case Mouse_Drag: | ||||||
|  |             { | ||||||
|  |                 if(0 /*event->buttons() & ~Qt::RightButton*/) | ||||||
|  |                 { | ||||||
|  |                     // cancel operation & return the object to the original position
 | ||||||
|  |                     placeObject(mGrabbedSceneNode, mOrigObjPos); | ||||||
|  |                     mMouseState = Mouse_Default; | ||||||
|  | 
 | ||||||
|  |                     // reset states
 | ||||||
|  |                     mCurrentMousePos = Ogre::Vector3(); | ||||||
|  |                     mOrigMousePos = Ogre::Vector3(); | ||||||
|  |                     mOrigObjPos = Ogre::Vector3(); | ||||||
|  |                     mGrabbedSceneNode = ""; | ||||||
|  |                     mCurrentObj = ""; | ||||||
|  |                     mOldPos = QPoint(0, 0); | ||||||
|  |                     mMouseEventTimer->invalidate(); | ||||||
|  |                     mOffset = 0.0f; | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             case Mouse_Edit: | ||||||
|  |             case Mouse_Default: | ||||||
|  |             { | ||||||
|  |                 if(event->buttons() & Qt::RightButton) | ||||||
|  |                 { | ||||||
|  |                     std::pair<std::string, Ogre::Vector3> result = objectUnderCursor(event->x(), event->y()); | ||||||
|  |                     if(result.first == "") | ||||||
|  |                         break; | ||||||
|  | 
 | ||||||
|  |                     mGrabbedSceneNode = result.first; | ||||||
|  |                     // ray test agaist the plane to get a starting position of the
 | ||||||
|  |                     // mouse in relation to the object position
 | ||||||
|  |                     //mPlane->redefine(Ogre::Vector3::UNIT_Z, result.second);
 | ||||||
|  |                     std::pair<Ogre::Vector3, Ogre::Vector3> planeRes = planeAxis(); | ||||||
|  |                     mPlane->redefine(planeRes.first, result.second); | ||||||
|  |                     std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); | ||||||
|  |                     if(planeResult.first) | ||||||
|  |                     { | ||||||
|  |                         mOrigMousePos = planeResult.second; | ||||||
|  |                         mCurrentMousePos = planeResult.second; | ||||||
|  |                         mOffset = 0.0f; | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     mOrigObjPos = mSceneManager->getSceneNode(mGrabbedSceneNode)->getPosition(); | ||||||
|  |                     mMouseEventTimer->start(); | ||||||
|  | 
 | ||||||
|  |                     mMouseState = Mouse_Grab; | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             /* NO_DEFAULT_CASE */ | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::mouseReleaseEvent (QMouseEvent *event) | ||||||
|  |     { | ||||||
|  |         switch(mMouseState) | ||||||
|  |         { | ||||||
|  |             case Mouse_Grab: | ||||||
|  |             { | ||||||
|  |                 std::pair<std::string, Ogre::Vector3> result = objectUnderCursor(event->x(), event->y()); | ||||||
|  |                 if(result.first != "") | ||||||
|  |                 { | ||||||
|  |                     if(result.first == mCurrentObj) | ||||||
|  |                     { | ||||||
|  |                         // unselect object
 | ||||||
|  |                         mMouseState = Mouse_Default; | ||||||
|  |                         mCurrentObj = ""; | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         // select object
 | ||||||
|  |                         mMouseState = Mouse_Edit; | ||||||
|  |                         mCurrentObj = result.first; | ||||||
|  | 
 | ||||||
|  |                         // print some debug info
 | ||||||
|  |                         if(isDebug()) | ||||||
|  |                         { | ||||||
|  |                             std::string referenceId = mPhysics->sceneNodeToRefId(result.first); | ||||||
|  |                             std::cout << "ReferenceId: " << referenceId << std::endl; | ||||||
|  |                             const CSMWorld::RefCollection& references = mParent->mDocument.getData().getReferences(); | ||||||
|  |                             int index = references.searchId(referenceId); | ||||||
|  |                             if (index != -1) | ||||||
|  |                             { | ||||||
|  |                                 int columnIndex = | ||||||
|  |                                     references.findColumnIndex(CSMWorld::Columns::ColumnId_ReferenceableId); | ||||||
|  |                                 std::cout << "  index: " + QString::number(index).toStdString() | ||||||
|  |                                           +", column index: " + QString::number(columnIndex).toStdString() | ||||||
|  |                                           << std::endl; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     // update highlighting the current object
 | ||||||
|  |                     if(isDebug()) | ||||||
|  |                         updateSelectionHighlight(result.first, result.second); | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             case Mouse_Drag: | ||||||
|  |             { | ||||||
|  |                 // final placement
 | ||||||
|  |                 std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); | ||||||
|  |                 if(planeResult.first) | ||||||
|  |                 { | ||||||
|  |                     if(mGrabbedSceneNode != "") | ||||||
|  |                     { | ||||||
|  |                         std::pair<Ogre::Vector3, Ogre::Vector3> planeRes = planeAxis(); | ||||||
|  |                         //mOrigObjPos.z += mOffset;
 | ||||||
|  |                         Ogre::Vector3 pos = mOrigObjPos+planeRes.first*mOffset+planeResult.second-mOrigMousePos; | ||||||
|  |                         placeObject(mGrabbedSceneNode, pos); | ||||||
|  |                         //mCurrentObj = mGrabbedSceneNode; // FIXME
 | ||||||
|  |                         mCurrentObj = "";                   // whether the object is selected
 | ||||||
|  | 
 | ||||||
|  |                         // reset states
 | ||||||
|  |                         mCurrentMousePos = Ogre::Vector3(); // mouse pos to use in wheel event
 | ||||||
|  |                         mOrigMousePos = Ogre::Vector3();    // starting pos of mouse in world space
 | ||||||
|  |                         mOrigObjPos = Ogre::Vector3();      // starting pos of object in world space
 | ||||||
|  |                         mGrabbedSceneNode = "";             // id of the object
 | ||||||
|  |                         mOffset = 0.0f;                    // used for z-axis movement
 | ||||||
|  |                         mOldPos = QPoint(0, 0);             // to calculate relative movement of mouse
 | ||||||
|  |                                                             //  on screen
 | ||||||
|  | 
 | ||||||
|  |                         // FIXME: update document
 | ||||||
|  |                         // FIXME: highlight current object?
 | ||||||
|  |                         mMouseState = Mouse_Edit; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             case Mouse_Edit: | ||||||
|  |             case Mouse_Default: | ||||||
|  |             { | ||||||
|  |                 // probably terrain, check
 | ||||||
|  |                 std::pair<std::string, Ogre::Vector3> result = terrainUnderCursor(event->x(), event->y()); | ||||||
|  |                 if(result.first != "") | ||||||
|  |                 { | ||||||
|  |                     if(isDebug()) | ||||||
|  |                     { | ||||||
|  |                         std::cout << "terrain: " << result.first << std::endl; | ||||||
|  |                         std::cout << "  hit pos "+ QString::number(result.second.x).toStdString() | ||||||
|  |                                 + ", " + QString::number(result.second.y).toStdString() | ||||||
|  |                                 + ", " + QString::number(result.second.z).toStdString() << std::endl; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             /* NO_DEFAULT_CASE */ | ||||||
|  |         } | ||||||
|  |         mMouseEventTimer->invalidate(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::mouseDoubleClickEvent (QMouseEvent *event) | ||||||
|  |     { | ||||||
|  |         if(0 && isDebug()) // disable
 | ||||||
|  |         { | ||||||
|  |             // FIXME: OEngine::PhysicEngine creates only one child scene node for the
 | ||||||
|  |             // debug drawer.  Hence only the first subview that creates the debug drawer
 | ||||||
|  |             // can view the debug lines.  Will need to keep a map in OEngine if multiple
 | ||||||
|  |             // subviews are to be supported.
 | ||||||
|  |             mPhysics->addSceneManager(mSceneManager, mParent); | ||||||
|  |             mPhysics->toggleDebugRendering(mSceneManager); | ||||||
|  |             mParent->flagAsModified(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     bool MouseState::wheelEvent (QWheelEvent *event) | ||||||
|  |     { | ||||||
|  |         switch(mMouseState) | ||||||
|  |         { | ||||||
|  |             case Mouse_Grab: | ||||||
|  |                 mMouseState = Mouse_Drag; | ||||||
|  | 
 | ||||||
|  |                 /* FALL_THROUGH */ | ||||||
|  |             case Mouse_Drag: | ||||||
|  |             { | ||||||
|  |                 // move the object along the z axis during Mouse_Drag or Mouse_Grab
 | ||||||
|  |                 if (event->delta()) | ||||||
|  |                 { | ||||||
|  |                     // seems positive is up and negative is down
 | ||||||
|  |                     mOffset += (event->delta()/1); // FIXME: arbitrary number, make config option?
 | ||||||
|  | 
 | ||||||
|  |                     std::pair<Ogre::Vector3, Ogre::Vector3> planeRes = planeAxis(); | ||||||
|  |                     Ogre::Vector3 pos = mOrigObjPos + planeRes.first*mOffset; | ||||||
|  |                     //pos.z += mOffset;
 | ||||||
|  |                     mSceneManager->getSceneNode(mGrabbedSceneNode)->setPosition(pos+mCurrentMousePos-mOrigMousePos); | ||||||
|  |                     mPhysics->moveSceneNodes(mGrabbedSceneNode, pos+mCurrentMousePos-mOrigMousePos); | ||||||
|  |                     updateSceneWidgets(); | ||||||
|  |                 } | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             case Mouse_Edit: | ||||||
|  |             case Mouse_Default: | ||||||
|  |             { | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |             /* NO_DEFAULT_CASE */ | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::pair<bool, Ogre::Vector3> MouseState::mousePositionOnPlane(const QPoint &pos, const Ogre::Plane &plane) | ||||||
|  |     { | ||||||
|  |         // using a really small value seems to mess up with the projections
 | ||||||
|  |         float nearClipDistance = getCamera()->getNearClipDistance(); // save existing
 | ||||||
|  |         getCamera()->setNearClipDistance(10.0f);  // arbitrary number
 | ||||||
|  |         Ogre::Ray mouseRay = getCamera()->getCameraToViewportRay( | ||||||
|  |             (float) pos.x() / getViewport()->getActualWidth(), | ||||||
|  |             (float) pos.y() / getViewport()->getActualHeight()); | ||||||
|  |         getCamera()->setNearClipDistance(nearClipDistance); // restore
 | ||||||
|  |         std::pair<bool, float> planeResult = mouseRay.intersects(plane); | ||||||
|  | 
 | ||||||
|  |         if(planeResult.first) | ||||||
|  |             return std::make_pair(true, mouseRay.getPoint(planeResult.second)); | ||||||
|  |         else | ||||||
|  |             return std::make_pair(false, Ogre::Vector3()); // should only happen if the plane is too small
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::pair<std::string, Ogre::Vector3> MouseState::terrainUnderCursor(const int mouseX, const int mouseY) | ||||||
|  |     { | ||||||
|  |         if(!getViewport()) | ||||||
|  |             return std::make_pair("", Ogre::Vector3()); | ||||||
|  | 
 | ||||||
|  |         float x = (float) mouseX / getViewport()->getActualWidth(); | ||||||
|  |         float y = (float) mouseY / getViewport()->getActualHeight(); | ||||||
|  | 
 | ||||||
|  |         std::pair<std::string, Ogre::Vector3> result = mPhysics->castRay(x, y, mSceneManager, getCamera()); | ||||||
|  |         if(result.first != "") | ||||||
|  |         { | ||||||
|  |             // FIXME: is there  a better way to distinguish terrain from objects?
 | ||||||
|  |             QString name  = QString(result.first.c_str()); | ||||||
|  |             if(name.contains(QRegExp("^HeightField"))) | ||||||
|  |             { | ||||||
|  |                 return result; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return std::make_pair("", Ogre::Vector3()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::pair<std::string, Ogre::Vector3> MouseState::objectUnderCursor(const int mouseX, const int mouseY) | ||||||
|  |     { | ||||||
|  |         if(!getViewport()) | ||||||
|  |             return std::make_pair("", Ogre::Vector3()); | ||||||
|  | 
 | ||||||
|  |         float x = (float) mouseX / getViewport()->getActualWidth(); | ||||||
|  |         float y = (float) mouseY / getViewport()->getActualHeight(); | ||||||
|  | 
 | ||||||
|  |         std::pair<std::string, Ogre::Vector3> result = mPhysics->castRay(x, y, mSceneManager, getCamera()); | ||||||
|  |         if(result.first != "") | ||||||
|  |         { | ||||||
|  |             // NOTE: anything not terrain is assumed to be an object
 | ||||||
|  |             QString name  = QString(result.first.c_str()); | ||||||
|  |             if(!name.contains(QRegExp("^HeightField"))) | ||||||
|  |             { | ||||||
|  |                 uint32_t visibilityMask = getViewport()->getVisibilityMask(); | ||||||
|  |                 bool ignoreObjects = !(visibilityMask & (uint32_t)CSVRender::Element_Reference); | ||||||
|  | 
 | ||||||
|  |                 if(!ignoreObjects && mSceneManager->hasSceneNode(result.first)) | ||||||
|  |                 { | ||||||
|  |                     return result; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return std::make_pair("", Ogre::Vector3()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // FIXME: for debugging only
 | ||||||
|  |     void MouseState::updateSelectionHighlight(const std::string sceneNode, const Ogre::Vector3 &position) | ||||||
|  |     { | ||||||
|  |         uint32_t visibilityMask = getViewport()->getVisibilityMask(); | ||||||
|  |         bool ignoreObjects = !(visibilityMask & (uint32_t)CSVRender::Element_Reference); | ||||||
|  | 
 | ||||||
|  |         if(ignoreObjects || !mSceneManager->hasSceneNode(sceneNode) || !isDebug()) | ||||||
|  |             return; | ||||||
|  | 
 | ||||||
|  |         CSMSettings::UserSettings &userSettings = CSMSettings::UserSettings::instance(); | ||||||
|  |         bool debugCursor = userSettings.setting( | ||||||
|  |                     "debug/mouse-position", QString("false")) == "true" ? true : false; | ||||||
|  | 
 | ||||||
|  |         //TODO: Try http://www.ogre3d.org/tikiwiki/Create+outline+around+a+character
 | ||||||
|  |         Ogre::SceneNode *scene = mSceneManager->getSceneNode(sceneNode); | ||||||
|  |         std::map<std::string, std::vector<std::string> >::iterator iter = | ||||||
|  |                                                 mSelectedEntities.find(sceneNode); | ||||||
|  |         if(iter != mSelectedEntities.end()) // currently selected
 | ||||||
|  |         { | ||||||
|  |             std::vector<std::string> clonedEntities = mSelectedEntities[sceneNode]; | ||||||
|  |             while(!clonedEntities.empty()) | ||||||
|  |             { | ||||||
|  |                 if(mSceneManager->hasEntity(clonedEntities.back())) | ||||||
|  |                 { | ||||||
|  |                     scene->detachObject(clonedEntities.back()); | ||||||
|  |                     mSceneManager->destroyEntity(clonedEntities.back()); | ||||||
|  |                 } | ||||||
|  |                 clonedEntities.pop_back(); | ||||||
|  |             } | ||||||
|  |             mSelectedEntities.erase(iter); | ||||||
|  | 
 | ||||||
|  |             if(debugCursor) | ||||||
|  |                 removeHitPoint(mSceneManager, sceneNode); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             std::vector<std::string> clonedEntities; | ||||||
|  |             Ogre::SceneNode::ObjectIterator iter = scene->getAttachedObjectIterator(); | ||||||
|  |             iter.begin(); | ||||||
|  |             while(iter.hasMoreElements()) | ||||||
|  |             { | ||||||
|  |                 Ogre::MovableObject * element = iter.getNext(); | ||||||
|  |                 if(!element) | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 if(element->getMovableType() != "Entity") | ||||||
|  |                     continue; | ||||||
|  | 
 | ||||||
|  |                 Ogre::Entity * entity = dynamic_cast<Ogre::Entity *>(element); | ||||||
|  |                 if(mSceneManager->hasEntity(entity->getName()+"cover")) | ||||||
|  |                 { | ||||||
|  |                     // FIXME: this shouldn't really happen... but does :(
 | ||||||
|  |                     scene->detachObject(entity->getName()+"cover"); | ||||||
|  |                     mSceneManager->destroyEntity(entity->getName()+"cover"); | ||||||
|  |                 } | ||||||
|  |                 Ogre::Entity * clone = entity->clone(entity->getName()+"cover"); | ||||||
|  | 
 | ||||||
|  |                 Ogre::MaterialPtr mat = | ||||||
|  |                     Ogre::MaterialManager::getSingleton().getByName("TransMaterial"); | ||||||
|  |                 if(!mat.isNull()) | ||||||
|  |                 { | ||||||
|  |                     clone->setMaterial(mat); | ||||||
|  |                     scene->attachObject(clone); | ||||||
|  |                     clonedEntities.push_back(entity->getName()+"cover"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             mSelectedEntities[sceneNode] = clonedEntities; | ||||||
|  | 
 | ||||||
|  |             if(debugCursor) | ||||||
|  |                 showHitPoint(mSceneManager, sceneNode, position); | ||||||
|  |         } | ||||||
|  |         mParent->flagAsModified(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::placeObject(const std::string sceneNode, const Ogre::Vector3 &pos) | ||||||
|  |     { | ||||||
|  |         mSceneManager->getSceneNode(sceneNode)->setPosition(pos); | ||||||
|  | 
 | ||||||
|  |         // update physics
 | ||||||
|  |         std::string refId = mPhysics->sceneNodeToRefId(sceneNode); | ||||||
|  |         const CSMWorld::CellRef& cellref = mParent->mDocument.getData().getReferences().getRecord (refId).get(); | ||||||
|  |         Ogre::Quaternion xr (Ogre::Radian (-cellref.mPos.rot[0]), Ogre::Vector3::UNIT_X); | ||||||
|  |         Ogre::Quaternion yr (Ogre::Radian (-cellref.mPos.rot[1]), Ogre::Vector3::UNIT_Y); | ||||||
|  |         Ogre::Quaternion zr (Ogre::Radian (-cellref.mPos.rot[2]), Ogre::Vector3::UNIT_Z); | ||||||
|  | 
 | ||||||
|  |         // FIXME: adjustRigidBody() seems to lose objects, work around by deleting and recreating objects
 | ||||||
|  |         //mPhysics->moveObject(sceneNode, pos, xr*yr*zr);
 | ||||||
|  |         mPhysics->replaceObject(sceneNode, refId, cellref.mScale, pos, xr*yr*zr); | ||||||
|  | 
 | ||||||
|  |         // update all SceneWidgets and their SceneManagers
 | ||||||
|  |         updateSceneWidgets(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void MouseState::updateSceneWidgets() | ||||||
|  |     { | ||||||
|  |         std::map<Ogre::SceneManager*, CSVRender::SceneWidget *> sceneWidgets = mPhysics->sceneWidgets(); | ||||||
|  | 
 | ||||||
|  |         std::map<Ogre::SceneManager*, CSVRender::SceneWidget *>::iterator iter = sceneWidgets.begin(); | ||||||
|  |         for(; iter != sceneWidgets.end(); ++iter) | ||||||
|  |         { | ||||||
|  |             (*iter).second->updateScene(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Ogre::Camera *MouseState::getCamera() | ||||||
|  |     { | ||||||
|  |         return mParent->getCamera(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Ogre::Viewport *MouseState::getViewport() | ||||||
|  |     { | ||||||
|  |         return mParent->getCamera()->getViewport(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     bool MouseState::isDebug() | ||||||
|  |     { | ||||||
|  |         CSMSettings::UserSettings &userSettings = CSMSettings::UserSettings::instance(); | ||||||
|  | 
 | ||||||
|  |         return userSettings.setting("debug/mouse-picking", QString("false")) == "true" ? true : false; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										82
									
								
								apps/opencs/view/render/mousestate.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								apps/opencs/view/render/mousestate.hpp
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,82 @@ | ||||||
|  | #ifndef OPENCS_VIEW_MOUSESTATE_H | ||||||
|  | #define OPENCS_VIEW_MOUSESTATE_H | ||||||
|  | 
 | ||||||
|  | #include <map> | ||||||
|  | #include <QPoint> | ||||||
|  | #include <OgreVector3.h> | ||||||
|  | 
 | ||||||
|  | class QElapsedTimer; | ||||||
|  | class QMouseEvent; | ||||||
|  | class QWheelEvent; | ||||||
|  | 
 | ||||||
|  | namespace Ogre | ||||||
|  | { | ||||||
|  |     class Plane; | ||||||
|  |     class SceneManager; | ||||||
|  |     class Camera; | ||||||
|  |     class Viewport; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | namespace CSVWorld | ||||||
|  | { | ||||||
|  |     class PhysicsSystem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | namespace CSVRender | ||||||
|  | { | ||||||
|  |     class WorldspaceWidget; | ||||||
|  | 
 | ||||||
|  |     class MouseState | ||||||
|  |     { | ||||||
|  |             enum MouseStates | ||||||
|  |             { | ||||||
|  |                 Mouse_Grab, | ||||||
|  |                 Mouse_Drag, | ||||||
|  |                 Mouse_Edit, | ||||||
|  |                 Mouse_Default | ||||||
|  |             }; | ||||||
|  |             MouseStates mMouseState; | ||||||
|  | 
 | ||||||
|  |             WorldspaceWidget *mParent; | ||||||
|  |             CSVWorld::PhysicsSystem *mPhysics; // local copy
 | ||||||
|  |             Ogre::SceneManager *mSceneManager; // local copy
 | ||||||
|  | 
 | ||||||
|  |             QPoint mOldPos; | ||||||
|  |             std::string mCurrentObj; | ||||||
|  |             std::string mGrabbedSceneNode; | ||||||
|  |             QElapsedTimer *mMouseEventTimer; | ||||||
|  |             Ogre::Plane *mPlane; | ||||||
|  |             Ogre::Vector3 mOrigObjPos; | ||||||
|  |             Ogre::Vector3 mOrigMousePos; | ||||||
|  |             Ogre::Vector3 mCurrentMousePos; | ||||||
|  |             float mOffset; | ||||||
|  | 
 | ||||||
|  |             std::map<std::string, std::vector<std::string> > mSelectedEntities; | ||||||
|  | 
 | ||||||
|  |         public: | ||||||
|  | 
 | ||||||
|  |             MouseState(WorldspaceWidget *parent); | ||||||
|  |             ~MouseState(); | ||||||
|  | 
 | ||||||
|  |             void mouseMoveEvent (QMouseEvent *event); | ||||||
|  |             void mousePressEvent (QMouseEvent *event); | ||||||
|  |             void mouseReleaseEvent (QMouseEvent *event); | ||||||
|  |             void mouseDoubleClickEvent (QMouseEvent *event); | ||||||
|  |             bool wheelEvent (QWheelEvent *event); | ||||||
|  | 
 | ||||||
|  |         private: | ||||||
|  | 
 | ||||||
|  |             std::pair<bool, Ogre::Vector3> mousePositionOnPlane(const QPoint &pos, const Ogre::Plane &plane); | ||||||
|  |             void placeObject(const std::string sceneNode, const Ogre::Vector3 &pos); | ||||||
|  |             std::pair<std::string, Ogre::Vector3> terrainUnderCursor(const int mouseX, const int mouseY); | ||||||
|  |             std::pair<std::string, Ogre::Vector3> objectUnderCursor(const int mouseX, const int mouseY); | ||||||
|  |             void updateSelectionHighlight(const std::string sceneNode, const Ogre::Vector3 &position); | ||||||
|  |             void updateSceneWidgets(); | ||||||
|  |             bool isDebug(); | ||||||
|  | 
 | ||||||
|  |             Ogre::Camera *getCamera();     // friend access
 | ||||||
|  |             Ogre::Viewport *getViewport(); // friend access
 | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif // OPENCS_VIEW_MOUSESTATE_H
 | ||||||
|  | @ -7,18 +7,10 @@ | ||||||
| #include <OgreSceneManager.h> | #include <OgreSceneManager.h> | ||||||
| #include <OgreEntity.h> | #include <OgreEntity.h> | ||||||
| 
 | 
 | ||||||
| #include <OgreMeshManager.h> |  | ||||||
| #include <OgreManualObject.h>        // FIXME: for debugging
 |  | ||||||
| #include <OgreMaterialManager.h>     // FIXME: for debugging
 |  | ||||||
| #include <OgreHardwarePixelBuffer.h> // FIXME: for debugging
 |  | ||||||
| 
 |  | ||||||
| #include <QMouseEvent> |  | ||||||
| #include <QElapsedTimer> |  | ||||||
| #include <QtGui/qevent.h> | #include <QtGui/qevent.h> | ||||||
| 
 | 
 | ||||||
| #include "../../model/world/universalid.hpp" | #include "../../model/world/universalid.hpp" | ||||||
| #include "../../model/world/idtable.hpp" | #include "../../model/world/idtable.hpp" | ||||||
| #include "../../model/settings/usersettings.hpp" |  | ||||||
| 
 | 
 | ||||||
| #include "../widget/scenetoolmode.hpp" | #include "../widget/scenetoolmode.hpp" | ||||||
| #include "../widget/scenetooltoggle.hpp" | #include "../widget/scenetooltoggle.hpp" | ||||||
|  | @ -27,96 +19,11 @@ | ||||||
| #include "../world/physicsmanager.hpp" | #include "../world/physicsmanager.hpp" | ||||||
| #include "../world/physicssystem.hpp" | #include "../world/physicssystem.hpp" | ||||||
| 
 | 
 | ||||||
|  | #include "mousestate.hpp" | ||||||
| #include "elements.hpp" | #include "elements.hpp" | ||||||
| 
 | 
 | ||||||
| namespace |  | ||||||
| { |  | ||||||
|     // FIXME: this section should be removed once the debugging is completed
 |  | ||||||
|     void showHitPoint(Ogre::SceneManager *sceneMgr, std::string name, Ogre::Vector3 point) |  | ||||||
|     { |  | ||||||
|         if(sceneMgr->hasManualObject("manual" + name)) |  | ||||||
|             sceneMgr->destroyManualObject("manual" + name); |  | ||||||
|         Ogre::ManualObject* manual = sceneMgr->createManualObject("manual" + name); |  | ||||||
|         manual->begin("BaseWhite", Ogre::RenderOperation::OT_LINE_LIST); |  | ||||||
|         manual-> position(point.x,     point.y,     point.z-100); |  | ||||||
|         manual-> position(point.x,     point.y,     point.z+100); |  | ||||||
|         manual-> position(point.x,     point.y-100, point.z); |  | ||||||
|         manual-> position(point.x,     point.y+100, point.z); |  | ||||||
|         manual-> position(point.x-100, point.y,     point.z); |  | ||||||
|         manual-> position(point.x+100, point.y,     point.z); |  | ||||||
|         manual->end(); |  | ||||||
|         sceneMgr->getRootSceneNode()->createChildSceneNode()->attachObject(manual); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void removeHitPoint(Ogre::SceneManager *sceneMgr, std::string name) |  | ||||||
|     { |  | ||||||
|         if(sceneMgr->hasManualObject("manual" + name)) |  | ||||||
|             sceneMgr->destroyManualObject("manual" + name); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void initDebug() |  | ||||||
|     { |  | ||||||
|         // material for visual cue on selected objects
 |  | ||||||
|         Ogre::TexturePtr texture = Ogre::TextureManager::getSingleton().getByName("DynamicTrans"); |  | ||||||
|         if(texture.isNull()) |  | ||||||
|         { |  | ||||||
|             texture = Ogre::TextureManager::getSingleton().createManual( |  | ||||||
|                 "DynamicTrans", // name
 |  | ||||||
|                 Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, |  | ||||||
|                 Ogre::TEX_TYPE_2D,  // type
 |  | ||||||
|                 8, 8,               // width & height
 |  | ||||||
|                 0,                  // number of mipmaps
 |  | ||||||
|                 Ogre::PF_BYTE_BGRA, // pixel format
 |  | ||||||
|                 Ogre::TU_DEFAULT);  // usage; should be TU_DYNAMIC_WRITE_ONLY_DISCARDABLE for
 |  | ||||||
|                                     // textures updated very often (e.g. each frame)
 |  | ||||||
| 
 |  | ||||||
|             Ogre::HardwarePixelBufferSharedPtr pixelBuffer = texture->getBuffer(); |  | ||||||
|             pixelBuffer->lock(Ogre::HardwareBuffer::HBL_NORMAL); |  | ||||||
|             const Ogre::PixelBox& pixelBox = pixelBuffer->getCurrentLock(); |  | ||||||
| 
 |  | ||||||
|             uint8_t* pDest = static_cast<uint8_t*>(pixelBox.data); |  | ||||||
| 
 |  | ||||||
|             // Fill in some pixel data. This will give a semi-transparent colour,
 |  | ||||||
|             // but this is of course dependent on the chosen pixel format.
 |  | ||||||
|             for (size_t j = 0; j < 8; j++) |  | ||||||
|             { |  | ||||||
|                 for(size_t i = 0; i < 8; i++) |  | ||||||
|                 { |  | ||||||
|                     *pDest++ = 255; // B
 |  | ||||||
|                     *pDest++ = 255; // G
 |  | ||||||
|                     *pDest++ = 127; // R
 |  | ||||||
|                     *pDest++ =  63; // A
 |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 pDest += pixelBox.getRowSkip() * Ogre::PixelUtil::getNumElemBytes(pixelBox.format); |  | ||||||
|             } |  | ||||||
|             pixelBuffer->unlock(); |  | ||||||
|         } |  | ||||||
|         Ogre::MaterialPtr material = Ogre::MaterialManager::getSingleton().getByName( |  | ||||||
|                     "TransMaterial"); |  | ||||||
|         if(material.isNull()) |  | ||||||
|         { |  | ||||||
|             Ogre::MaterialPtr material = Ogre::MaterialManager::getSingleton().create( |  | ||||||
|                         "TransMaterial", |  | ||||||
|                         Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, true ); |  | ||||||
|             Ogre::Pass *pass = material->getTechnique( 0 )->getPass( 0 ); |  | ||||||
|             pass->setLightingEnabled( false ); |  | ||||||
|             pass->setDepthWriteEnabled( false ); |  | ||||||
|             pass->setSceneBlending( Ogre::SBT_TRANSPARENT_ALPHA ); |  | ||||||
| 
 |  | ||||||
|             Ogre::TextureUnitState *tex = pass->createTextureUnitState("CustomState", 0); |  | ||||||
|             tex->setTextureName("DynamicTrans"); |  | ||||||
|             tex->setTextureFiltering( Ogre::TFO_ANISOTROPIC ); |  | ||||||
|             material->load(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| CSVRender::WorldspaceWidget::WorldspaceWidget (CSMDoc::Document& document, QWidget* parent) | CSVRender::WorldspaceWidget::WorldspaceWidget (CSMDoc::Document& document, QWidget* parent) | ||||||
| : SceneWidget (parent), mDocument(document), mSceneElements(0), mRun(0), | : SceneWidget (parent), mDocument(document), mSceneElements(0), mRun(0), mPhysics(0), mMouse(0) | ||||||
|   mCurrentObj(""), mMouseState(Mouse_Default), mOldPos(0,0), mMouseEventTimer(0), mPlane(0), |  | ||||||
|   mGrabbedSceneNode(""), mOrigObjPos(Ogre::Vector3()), mOrigMousePos(Ogre::Vector3()), |  | ||||||
|   mCurrentMousePos(Ogre::Vector3()), mZOffset(0.0f) |  | ||||||
| { | { | ||||||
|     setAcceptDrops(true); |     setAcceptDrops(true); | ||||||
| 
 | 
 | ||||||
|  | @ -148,53 +55,18 @@ CSVRender::WorldspaceWidget::WorldspaceWidget (CSMDoc::Document& document, QWidg | ||||||
|     connect (debugProfiles, SIGNAL (rowsAboutToBeRemoved (const QModelIndex&, int, int)), |     connect (debugProfiles, SIGNAL (rowsAboutToBeRemoved (const QModelIndex&, int, int)), | ||||||
|         this, SLOT (debugProfileAboutToBeRemoved (const QModelIndex&, int, int))); |         this, SLOT (debugProfileAboutToBeRemoved (const QModelIndex&, int, int))); | ||||||
| 
 | 
 | ||||||
|     // associate WorldSpaceWidgets (and their SceneManagers with Documents, then create physics if new document
 |     // associate WorldSpaceWidgets (and their SceneManagers) with Documents
 | ||||||
|  |     // then create physics if there is a new document
 | ||||||
|     mPhysics = CSVWorld::PhysicsManager::instance()->addSceneWidget(document, this); |     mPhysics = CSVWorld::PhysicsManager::instance()->addSceneWidget(document, this); | ||||||
|     mPhysics->addSceneManager(getSceneManager(), this); |     mPhysics->addSceneManager(getSceneManager(), this); | ||||||
| 
 |     mMouse = new MouseState(this); | ||||||
|     initDebug(); |  | ||||||
|     mMouseEventTimer = new QElapsedTimer(); |  | ||||||
|     mMouseEventTimer->invalidate(); |  | ||||||
| 
 |  | ||||||
|     mPlane = new Ogre::Plane(Ogre::Vector3::UNIT_Z, 0); |  | ||||||
|     Ogre::MeshPtr mesh = Ogre::MeshManager::getSingleton().createPlane("ground", |  | ||||||
|         Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, |  | ||||||
|         *mPlane, |  | ||||||
|         300000,300000, // FIXME: use far clip dist?
 |  | ||||||
|         1,1, // segments
 |  | ||||||
|         true,  // normals
 |  | ||||||
|         1,     // numTexCoordSets
 |  | ||||||
|         1,1, // uTile, vTile
 |  | ||||||
|         Ogre::Vector3::UNIT_Y // upVector
 |  | ||||||
|         ); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| CSVRender::WorldspaceWidget::~WorldspaceWidget () | CSVRender::WorldspaceWidget::~WorldspaceWidget () | ||||||
| { | { | ||||||
|  |     delete mMouse; | ||||||
|     mPhysics->removeSceneManager(getSceneManager()); |     mPhysics->removeSceneManager(getSceneManager()); | ||||||
|     CSVWorld::PhysicsManager::instance()->removeSceneWidget(this); |     CSVWorld::PhysicsManager::instance()->removeSceneWidget(this); | ||||||
| 
 |  | ||||||
|     delete mMouseEventTimer; |  | ||||||
| 
 |  | ||||||
|     // For debugging only
 |  | ||||||
|     std::map<std::string, std::vector<std::string> >::iterator iter = mSelectedEntities.begin(); |  | ||||||
|     for(;iter != mSelectedEntities.end(); ++iter) |  | ||||||
|     { |  | ||||||
|         removeHitPoint(getSceneManager(), iter->first); |  | ||||||
| 
 |  | ||||||
|         if(getSceneManager()->hasSceneNode(iter->first)) |  | ||||||
|         { |  | ||||||
|             Ogre::SceneNode *scene = getSceneManager()->getSceneNode(iter->first); |  | ||||||
| 
 |  | ||||||
|             if(scene) |  | ||||||
|             { |  | ||||||
|                 scene->removeAndDestroyAllChildren(); |  | ||||||
|                 getSceneManager()->destroySceneNode(iter->first); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     delete mPlane; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void CSVRender::WorldspaceWidget::selectNavigationMode (const std::string& mode) | void CSVRender::WorldspaceWidget::selectNavigationMode (const std::string& mode) | ||||||
|  | @ -471,147 +343,18 @@ CSVWorld::PhysicsSystem *CSVRender::WorldspaceWidget::getPhysics() | ||||||
|     return mPhysics; |     return mPhysics; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // mouse picking
 |  | ||||||
| // FIXME: need to virtualise mouse buttons
 |  | ||||||
| //
 |  | ||||||
| // State machine:
 |  | ||||||
| //
 |  | ||||||
| // [default] mousePressEvent->check if the mouse is pointing at an object
 |  | ||||||
| //         if yes, create collision planes then go to [grab]
 |  | ||||||
| //         else check for terrain
 |  | ||||||
| //
 |  | ||||||
| // [grab]  mouseReleaseEvent->if same button and new obj, go to [edit]
 |  | ||||||
| //         mouseMoveEvent->if same button, go to [drag]
 |  | ||||||
| //         other mouse events or buttons, go back to [default] (i.e. like 'cancel')
 |  | ||||||
| //
 |  | ||||||
| // [drag]  mouseReleaseEvent->if same button, place the object at the new
 |  | ||||||
| //         location, update the document then go to [edit]
 |  | ||||||
| //         mouseMoveEvent->update position to the user based on ray to the collision
 |  | ||||||
| //         planes and render the object at the new location, but do not update
 |  | ||||||
| //         the document yet
 |  | ||||||
| //
 |  | ||||||
| // [edit]  TODO, probably fine positional adjustments or rotations; clone/delete?
 |  | ||||||
| //
 |  | ||||||
| //
 |  | ||||||
| //               press               press (obj)
 |  | ||||||
| //   [default] --------> [grab] <-------------------- [edit]
 |  | ||||||
| //       ^       (obj)    |  |  ------> [drag] ----->   ^
 |  | ||||||
| //       |                |  |   move     ^ |  release  |
 |  | ||||||
| //       |                |  |            | |           |
 |  | ||||||
| //       |                |  |            +-+           |
 |  | ||||||
| //       |                |  |            move          |
 |  | ||||||
| //       +----------------+  +--------------------------+
 |  | ||||||
| //            release                  release
 |  | ||||||
| //           (same obj)               (new obj)
 |  | ||||||
| //
 |  | ||||||
| //
 |  | ||||||
| void CSVRender::WorldspaceWidget::mouseMoveEvent (QMouseEvent *event) | void CSVRender::WorldspaceWidget::mouseMoveEvent (QMouseEvent *event) | ||||||
| { | { | ||||||
|     if(event->buttons() & Qt::RightButton) |     if(event->buttons() & Qt::RightButton) | ||||||
|     { |     { | ||||||
|         switch(mMouseState) |         mMouse->mouseMoveEvent(event); | ||||||
|         { |  | ||||||
|             case Mouse_Grab: |  | ||||||
|             { |  | ||||||
|                 // check if min elapsed time to stop false detection of drag
 |  | ||||||
|                 if(!mMouseEventTimer->isValid() || !mMouseEventTimer->hasExpired(100)) // ms
 |  | ||||||
|                     break; |  | ||||||
| 
 |  | ||||||
|                 mMouseEventTimer->invalidate(); |  | ||||||
|                 mMouseState = Mouse_Drag; |  | ||||||
| 
 |  | ||||||
|                 /* FALL_THROUGH */ |  | ||||||
|             } |  | ||||||
|             case Mouse_Drag: |  | ||||||
|             { |  | ||||||
|                 if(event->pos() != mOldPos) // TODO: maybe don't update less than a quantum?
 |  | ||||||
|                 { |  | ||||||
|                     mOldPos = event->pos(); |  | ||||||
| 
 |  | ||||||
|                     // ray test against the plane to provide feedback to the user the
 |  | ||||||
|                     // relative movement of the object on the x-y plane
 |  | ||||||
|                     std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); |  | ||||||
|                     if(planeResult.first) |  | ||||||
|                     { |  | ||||||
|                         if(mGrabbedSceneNode != "") |  | ||||||
|                         { |  | ||||||
|                             Ogre::Vector3 pos = mOrigObjPos; |  | ||||||
|                             pos.z += mZOffset; |  | ||||||
|                             getSceneManager()->getSceneNode(mGrabbedSceneNode)->setPosition(pos+planeResult.second-mOrigMousePos); |  | ||||||
|                             mCurrentMousePos = planeResult.second; |  | ||||||
|                             mPhysics->moveSceneNodes(mGrabbedSceneNode, pos+planeResult.second-mOrigMousePos); |  | ||||||
|                             updateSceneWidgets(); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|             case Mouse_Edit: |  | ||||||
|             case Mouse_Default: |  | ||||||
|             { |  | ||||||
|                 break; // error event, ignore
 |  | ||||||
|             } |  | ||||||
|             /* NO_DEFAULT_CASE */ |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|     SceneWidget::mouseMoveEvent(event); |     SceneWidget::mouseMoveEvent(event); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void CSVRender::WorldspaceWidget::mousePressEvent (QMouseEvent *event) | void CSVRender::WorldspaceWidget::mousePressEvent (QMouseEvent *event) | ||||||
| { | { | ||||||
|     switch(mMouseState) |     mMouse->mousePressEvent(event); | ||||||
|     { |  | ||||||
|         case Mouse_Grab: |  | ||||||
|         case Mouse_Drag: |  | ||||||
|         { |  | ||||||
|             if(0 /*event->buttons() & ~Qt::RightButton*/) |  | ||||||
|             { |  | ||||||
|                 // cancel operation & return the object to the original position
 |  | ||||||
|                 placeObject(mGrabbedSceneNode, mOrigObjPos); |  | ||||||
|                 mMouseState = Mouse_Default; |  | ||||||
| 
 |  | ||||||
|                 // reset states
 |  | ||||||
|                 mCurrentMousePos = Ogre::Vector3(); |  | ||||||
|                 mOrigMousePos = Ogre::Vector3(); |  | ||||||
|                 mOrigObjPos = Ogre::Vector3(); |  | ||||||
|                 mGrabbedSceneNode = ""; |  | ||||||
|                 mCurrentObj = ""; |  | ||||||
|                 mOldPos = QPoint(0, 0); |  | ||||||
|                 mMouseEventTimer->invalidate(); |  | ||||||
|                 mZOffset = 0.0f; |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case Mouse_Edit: |  | ||||||
|         case Mouse_Default: |  | ||||||
|         { |  | ||||||
|             if(event->buttons() & Qt::RightButton) |  | ||||||
|             { |  | ||||||
|                 std::pair<std::string, Ogre::Vector3> result = objectUnderCursor(event->x(), event->y()); |  | ||||||
|                 if(result.first == "") |  | ||||||
|                     break; |  | ||||||
| 
 |  | ||||||
|                 mGrabbedSceneNode = result.first; |  | ||||||
|                 // ray test agaist the plane to get a starting position of the
 |  | ||||||
|                 // mouse in relation to the object position
 |  | ||||||
|                 mPlane->redefine(Ogre::Vector3::UNIT_Z, result.second); |  | ||||||
|                 std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); |  | ||||||
|                 if(planeResult.first) |  | ||||||
|                 { |  | ||||||
|                     mOrigMousePos = planeResult.second; |  | ||||||
|                     mCurrentMousePos = planeResult.second; |  | ||||||
|                     mZOffset = 0.0f; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 mOrigObjPos = getSceneManager()->getSceneNode(mGrabbedSceneNode)->getPosition(); |  | ||||||
|                 mMouseEventTimer->start(); |  | ||||||
| 
 |  | ||||||
|                 mMouseState = Mouse_Grab; |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         /* NO_DEFAULT_CASE */ |  | ||||||
|     } |  | ||||||
|     //SceneWidget::mousePressEvent(event);
 |     //SceneWidget::mousePressEvent(event);
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -619,104 +362,12 @@ void CSVRender::WorldspaceWidget::mouseReleaseEvent (QMouseEvent *event) | ||||||
| { | { | ||||||
|     if(event->button() == Qt::RightButton) |     if(event->button() == Qt::RightButton) | ||||||
|     { |     { | ||||||
|         if(!getCamera()->getViewport()) |         if(!getViewport()) | ||||||
|         { |         { | ||||||
|             SceneWidget::mouseReleaseEvent(event); |             SceneWidget::mouseReleaseEvent(event); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 |         mMouse->mouseReleaseEvent(event); | ||||||
|         switch(mMouseState) |  | ||||||
|         { |  | ||||||
|             case Mouse_Grab: |  | ||||||
|             { |  | ||||||
|                 std::pair<std::string, Ogre::Vector3> result = objectUnderCursor(event->x(), event->y()); |  | ||||||
|                 if(result.first != "") |  | ||||||
|                 { |  | ||||||
|                     if(result.first == mCurrentObj) |  | ||||||
|                     { |  | ||||||
|                         // unselect object
 |  | ||||||
|                         mMouseState = Mouse_Default; |  | ||||||
|                         mCurrentObj = ""; |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         // select object
 |  | ||||||
|                         mMouseState = Mouse_Edit; |  | ||||||
|                         mCurrentObj = result.first; |  | ||||||
| 
 |  | ||||||
|                         // print some debug info
 |  | ||||||
|                         if(isDebug()) |  | ||||||
|                         { |  | ||||||
|                             std::string referenceId = mPhysics->sceneNodeToRefId(result.first); |  | ||||||
|                             std::cout << "ReferenceId: " << referenceId << std::endl; |  | ||||||
|                             const CSMWorld::RefCollection& references = mDocument.getData().getReferences(); |  | ||||||
|                             int index = references.searchId(referenceId); |  | ||||||
|                             if (index != -1) |  | ||||||
|                             { |  | ||||||
|                                 int columnIndex = |  | ||||||
|                                     references.findColumnIndex(CSMWorld::Columns::ColumnId_ReferenceableId); |  | ||||||
|                                 std::cout << "  index: " + QString::number(index).toStdString() |  | ||||||
|                                           +", column index: " + QString::number(columnIndex).toStdString() |  | ||||||
|                                           << std::endl; |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     // update highlighting the current object
 |  | ||||||
|                     if(isDebug()) |  | ||||||
|                         updateSelectionHighlight(result.first, result.second); |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|             case Mouse_Drag: |  | ||||||
|             { |  | ||||||
|                 // final placement
 |  | ||||||
|                 std::pair<bool, Ogre::Vector3> planeResult = mousePositionOnPlane(event->pos(), *mPlane); |  | ||||||
|                 if(planeResult.first) |  | ||||||
|                 { |  | ||||||
|                     if(mGrabbedSceneNode != "") |  | ||||||
|                     { |  | ||||||
|                         mOrigObjPos.z += mZOffset; |  | ||||||
|                         Ogre::Vector3 pos = mOrigObjPos+planeResult.second-mOrigMousePos; |  | ||||||
|                         placeObject(mGrabbedSceneNode, pos); |  | ||||||
|                         //mCurrentObj = mGrabbedSceneNode; // FIXME
 |  | ||||||
|                         mCurrentObj = "";                   // whether the object is selected
 |  | ||||||
| 
 |  | ||||||
|                         // reset states
 |  | ||||||
|                         mCurrentMousePos = Ogre::Vector3(); // mouse pos to use in wheel event
 |  | ||||||
|                         mOrigMousePos = Ogre::Vector3();    // starting pos of mouse in world space
 |  | ||||||
|                         mOrigObjPos = Ogre::Vector3();      // starting pos of object in world space
 |  | ||||||
|                         mGrabbedSceneNode = "";             // id of the object
 |  | ||||||
|                         mZOffset = 0.0f;                    // used for z-axis movement
 |  | ||||||
|                         mOldPos = QPoint(0, 0);             // to calculate relative movement of mouse
 |  | ||||||
|                                                             //  on screen
 |  | ||||||
| 
 |  | ||||||
|                         // FIXME: update document
 |  | ||||||
|                         // FIXME: highlight current object?
 |  | ||||||
|                         mMouseState = Mouse_Edit; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|             case Mouse_Edit: |  | ||||||
|             case Mouse_Default: |  | ||||||
|             { |  | ||||||
|                 // probably terrain, check
 |  | ||||||
|                 std::pair<std::string, Ogre::Vector3> result = terrainUnderCursor(event->x(), event->y()); |  | ||||||
|                 if(result.first != "") |  | ||||||
|                 { |  | ||||||
|                     if(isDebug()) |  | ||||||
|                     { |  | ||||||
|                         std::cout << "terrain: " << result.first << std::endl; |  | ||||||
|                         std::cout << "  hit pos "+ QString::number(result.second.x).toStdString() |  | ||||||
|                                 + ", " + QString::number(result.second.y).toStdString() |  | ||||||
|                                 + ", " + QString::number(result.second.z).toStdString() << std::endl; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|             /* NO_DEFAULT_CASE */ |  | ||||||
|         } |  | ||||||
|         mMouseEventTimer->invalidate(); |  | ||||||
|     } |     } | ||||||
|     SceneWidget::mouseReleaseEvent(event); |     SceneWidget::mouseReleaseEvent(event); | ||||||
| } | } | ||||||
|  | @ -725,226 +376,13 @@ void CSVRender::WorldspaceWidget::mouseDoubleClickEvent (QMouseEvent *event) | ||||||
| { | { | ||||||
|     if(event->button() == Qt::RightButton) |     if(event->button() == Qt::RightButton) | ||||||
|     { |     { | ||||||
|         if(isDebug()) |         mMouse->mouseDoubleClickEvent(event); | ||||||
|         { |  | ||||||
|             // FIXME: OEngine::PhysicEngine creates only one child scene node for the
 |  | ||||||
|             // debug drawer.  Hence only the first subview that creates the debug drawer
 |  | ||||||
|             // can view the debug lines.  Will need to keep a map in OEngine if multiple
 |  | ||||||
|             // subviews are to be supported.
 |  | ||||||
|             //mPhysics->setSceneManager(getSceneManager());
 |  | ||||||
|             mPhysics->toggleDebugRendering(getSceneManager()); |  | ||||||
|             flagAsModified(); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|     //SceneWidget::mouseDoubleClickEvent(event);
 |     //SceneWidget::mouseDoubleClickEvent(event);
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void CSVRender::WorldspaceWidget::wheelEvent (QWheelEvent *event) | void CSVRender::WorldspaceWidget::wheelEvent (QWheelEvent *event) | ||||||
| { | { | ||||||
|     switch(mMouseState) |     if(!mMouse->wheelEvent(event)) | ||||||
|     { |  | ||||||
|         case Mouse_Grab: |  | ||||||
|             mMouseState = Mouse_Drag; |  | ||||||
| 
 |  | ||||||
|             /* FALL_THROUGH */ |  | ||||||
|         case Mouse_Drag: |  | ||||||
|         { |  | ||||||
|             // move the object along the z axis during Mouse_Drag or Mouse_Grab
 |  | ||||||
|             if (event->delta()) |  | ||||||
|             { |  | ||||||
|                 // seems positive is up and negative is down
 |  | ||||||
|                 mZOffset += (event->delta()/5); // FIXME: arbitrary number, make config option?
 |  | ||||||
| 
 |  | ||||||
|                 Ogre::Vector3 pos = mOrigObjPos; |  | ||||||
|                 pos.z += mZOffset; |  | ||||||
|                 getSceneManager()->getSceneNode(mGrabbedSceneNode)->setPosition(pos+mCurrentMousePos-mOrigMousePos); |  | ||||||
|                 mPhysics->moveSceneNodes(mGrabbedSceneNode, pos+mCurrentMousePos-mOrigMousePos); |  | ||||||
|                 updateSceneWidgets(); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case Mouse_Edit: |  | ||||||
|         case Mouse_Default: |  | ||||||
|         { |  | ||||||
|         SceneWidget::wheelEvent(event); |         SceneWidget::wheelEvent(event); | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         /* NO_DEFAULT_CASE */ |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // FIXME: for debugging only
 |  | ||||||
| void CSVRender::WorldspaceWidget::updateSelectionHighlight(const std::string sceneNode, const Ogre::Vector3 &position) |  | ||||||
| { |  | ||||||
|     uint32_t visibilityMask = getCamera()->getViewport()->getVisibilityMask(); |  | ||||||
|     bool ignoreObjects = !(visibilityMask & (uint32_t)CSVRender::Element_Reference); |  | ||||||
| 
 |  | ||||||
|     if(ignoreObjects || !getSceneManager()->hasSceneNode(sceneNode) || !isDebug()) |  | ||||||
|         return; |  | ||||||
| 
 |  | ||||||
|     CSMSettings::UserSettings &userSettings = CSMSettings::UserSettings::instance(); |  | ||||||
|     bool debugCursor = userSettings.setting( |  | ||||||
|                 "debug/mouse-position", QString("false")) == "true" ? true : false; |  | ||||||
| 
 |  | ||||||
|     //TODO: Try http://www.ogre3d.org/tikiwiki/Create+outline+around+a+character
 |  | ||||||
|     Ogre::SceneNode *scene = getSceneManager()->getSceneNode(sceneNode); |  | ||||||
|     std::map<std::string, std::vector<std::string> >::iterator iter = |  | ||||||
|                                             mSelectedEntities.find(sceneNode); |  | ||||||
|     if(iter != mSelectedEntities.end()) // currently selected
 |  | ||||||
|     { |  | ||||||
|         std::vector<std::string> clonedEntities = mSelectedEntities[sceneNode]; |  | ||||||
|         while(!clonedEntities.empty()) |  | ||||||
|         { |  | ||||||
|             if(getSceneManager()->hasEntity(clonedEntities.back())) |  | ||||||
|             { |  | ||||||
|                 scene->detachObject(clonedEntities.back()); |  | ||||||
|                 getSceneManager()->destroyEntity(clonedEntities.back()); |  | ||||||
|             } |  | ||||||
|             clonedEntities.pop_back(); |  | ||||||
|         } |  | ||||||
|         mSelectedEntities.erase(iter); |  | ||||||
| 
 |  | ||||||
|         if(debugCursor) |  | ||||||
|             removeHitPoint(getSceneManager(), sceneNode); |  | ||||||
|     } |  | ||||||
|     else |  | ||||||
|     { |  | ||||||
|         std::vector<std::string> clonedEntities; |  | ||||||
|         Ogre::SceneNode::ObjectIterator iter = scene->getAttachedObjectIterator(); |  | ||||||
|         iter.begin(); |  | ||||||
|         while(iter.hasMoreElements()) |  | ||||||
|         { |  | ||||||
|             Ogre::MovableObject * element = iter.getNext(); |  | ||||||
|             if(!element) |  | ||||||
|                 break; |  | ||||||
| 
 |  | ||||||
|             if(element->getMovableType() != "Entity") |  | ||||||
|                 continue; |  | ||||||
| 
 |  | ||||||
|             Ogre::Entity * entity = dynamic_cast<Ogre::Entity *>(element); |  | ||||||
|             if(getSceneManager()->hasEntity(entity->getName()+"cover")) |  | ||||||
|             { |  | ||||||
|                 // FIXME: this shouldn't really happen... but does :(
 |  | ||||||
|                 scene->detachObject(entity->getName()+"cover"); |  | ||||||
|                 getSceneManager()->destroyEntity(entity->getName()+"cover"); |  | ||||||
|             } |  | ||||||
|             Ogre::Entity * clone = entity->clone(entity->getName()+"cover"); |  | ||||||
| 
 |  | ||||||
|             Ogre::MaterialPtr mat = |  | ||||||
|                 Ogre::MaterialManager::getSingleton().getByName("TransMaterial"); |  | ||||||
|             if(!mat.isNull()) |  | ||||||
|             { |  | ||||||
|                 clone->setMaterial(mat); |  | ||||||
|                 scene->attachObject(clone); |  | ||||||
|                 clonedEntities.push_back(entity->getName()+"cover"); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         mSelectedEntities[sceneNode] = clonedEntities; |  | ||||||
| 
 |  | ||||||
|         if(debugCursor) |  | ||||||
|             showHitPoint(getSceneManager(), sceneNode, position); |  | ||||||
|     } |  | ||||||
|     flagAsModified(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::pair<std::string, Ogre::Vector3> CSVRender::WorldspaceWidget::terrainUnderCursor(const int mouseX, const int mouseY) |  | ||||||
| { |  | ||||||
|     if(!getCamera()->getViewport()) |  | ||||||
|         return std::make_pair("", Ogre::Vector3()); |  | ||||||
| 
 |  | ||||||
|     float x = (float) mouseX / getCamera()->getViewport()->getActualWidth(); |  | ||||||
|     float y = (float) mouseY / getCamera()->getViewport()->getActualHeight(); |  | ||||||
| 
 |  | ||||||
|     std::pair<std::string, Ogre::Vector3> result = mPhysics->castRay(x, y, getSceneManager(), getCamera()); |  | ||||||
|     if(result.first != "") |  | ||||||
|     { |  | ||||||
|         // FIXME: is there  a better way to distinguish terrain from objects?
 |  | ||||||
|         QString name  = QString(result.first.c_str()); |  | ||||||
|         if(name.contains(QRegExp("^HeightField"))) |  | ||||||
|         { |  | ||||||
|             return result; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return std::make_pair("", Ogre::Vector3()); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::pair<std::string, Ogre::Vector3> CSVRender::WorldspaceWidget::objectUnderCursor(const int mouseX, const int mouseY) |  | ||||||
| { |  | ||||||
|     if(!getCamera()->getViewport()) |  | ||||||
|         return std::make_pair("", Ogre::Vector3()); |  | ||||||
| 
 |  | ||||||
|     float x = (float) mouseX / getCamera()->getViewport()->getActualWidth(); |  | ||||||
|     float y = (float) mouseY / getCamera()->getViewport()->getActualHeight(); |  | ||||||
| 
 |  | ||||||
|     std::pair<std::string, Ogre::Vector3> result = mPhysics->castRay(x, y, getSceneManager(), getCamera()); |  | ||||||
|     if(result.first != "") |  | ||||||
|     { |  | ||||||
|         // NOTE: anything not terrain is assumed to be an object
 |  | ||||||
|         QString name  = QString(result.first.c_str()); |  | ||||||
|         if(!name.contains(QRegExp("^HeightField"))) |  | ||||||
|         { |  | ||||||
|             uint32_t visibilityMask = getCamera()->getViewport()->getVisibilityMask(); |  | ||||||
|             bool ignoreObjects = !(visibilityMask & (uint32_t)CSVRender::Element_Reference); |  | ||||||
| 
 |  | ||||||
|             if(!ignoreObjects && getSceneManager()->hasSceneNode(result.first)) |  | ||||||
|             { |  | ||||||
|                 return result; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return std::make_pair("", Ogre::Vector3()); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::pair<bool, Ogre::Vector3> CSVRender::WorldspaceWidget::mousePositionOnPlane(const QPoint &pos, const Ogre::Plane &plane) |  | ||||||
| { |  | ||||||
|     // using a really small value seems to mess up with the projections
 |  | ||||||
|     float nearClipDistance = getCamera()->getNearClipDistance(); // save existing
 |  | ||||||
|     getCamera()->setNearClipDistance(10.0f);  // arbitrary number
 |  | ||||||
|     Ogre::Ray mouseRay = getCamera()->getCameraToViewportRay( |  | ||||||
|         (float) pos.x() / getCamera()->getViewport()->getActualWidth(), |  | ||||||
|         (float) pos.y() / getCamera()->getViewport()->getActualHeight()); |  | ||||||
|     getCamera()->setNearClipDistance(nearClipDistance); // restore
 |  | ||||||
|     std::pair<bool, float> planeResult = mouseRay.intersects(plane); |  | ||||||
| 
 |  | ||||||
|     if(planeResult.first) |  | ||||||
|         return std::make_pair(true, mouseRay.getPoint(planeResult.second)); |  | ||||||
|     else |  | ||||||
|         return std::make_pair(false, Ogre::Vector3()); // should only happen if the plane is too small
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void CSVRender::WorldspaceWidget::placeObject(const std::string sceneNode, const Ogre::Vector3 &pos) |  | ||||||
| { |  | ||||||
|     getSceneManager()->getSceneNode(sceneNode)->setPosition(pos); |  | ||||||
| 
 |  | ||||||
|     // update physics
 |  | ||||||
|     std::string refId = mPhysics->sceneNodeToRefId(sceneNode); |  | ||||||
|     const CSMWorld::CellRef& cellref = mDocument.getData().getReferences().getRecord (refId).get(); |  | ||||||
|     Ogre::Quaternion xr (Ogre::Radian (-cellref.mPos.rot[0]), Ogre::Vector3::UNIT_X); |  | ||||||
|     Ogre::Quaternion yr (Ogre::Radian (-cellref.mPos.rot[1]), Ogre::Vector3::UNIT_Y); |  | ||||||
|     Ogre::Quaternion zr (Ogre::Radian (-cellref.mPos.rot[2]), Ogre::Vector3::UNIT_Z); |  | ||||||
| 
 |  | ||||||
|     // FIXME: adjustRigidBody() seems to lose objects, work around by deleting and recreating objects
 |  | ||||||
|     //mPhysics->moveObject(sceneNode, pos, xr*yr*zr);
 |  | ||||||
|     mPhysics->replaceObject(sceneNode, refId, cellref.mScale, pos, xr*yr*zr); |  | ||||||
| 
 |  | ||||||
|     // update all SceneWidgets and their SceneManagers
 |  | ||||||
|     updateSceneWidgets(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void CSVRender::WorldspaceWidget::updateSceneWidgets() |  | ||||||
| { |  | ||||||
|     std::map<Ogre::SceneManager*, CSVRender::SceneWidget *> sceneWidgets = mPhysics->sceneWidgets(); |  | ||||||
| 
 |  | ||||||
|     std::map<Ogre::SceneManager*, CSVRender::SceneWidget *>::iterator iter = sceneWidgets.begin(); |  | ||||||
|     for(; iter != sceneWidgets.end(); ++iter) |  | ||||||
|     { |  | ||||||
|         (*iter).second->updateScene(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| bool CSVRender::WorldspaceWidget::isDebug() |  | ||||||
| { |  | ||||||
|     return false; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -27,12 +27,12 @@ namespace CSVWorld | ||||||
|     class PhysicsSystem; |     class PhysicsSystem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| class QElapsedTimer; |  | ||||||
| 
 |  | ||||||
| namespace CSVRender | namespace CSVRender | ||||||
| { | { | ||||||
|     class WorldspaceWidget : public SceneWidget |     class WorldspaceWidget : public SceneWidget | ||||||
|     { |     { | ||||||
|  |         friend class MouseState; | ||||||
|  | 
 | ||||||
|             Q_OBJECT |             Q_OBJECT | ||||||
| 
 | 
 | ||||||
|             CSVRender::Navigation1st m1st; |             CSVRender::Navigation1st m1st; | ||||||
|  | @ -42,26 +42,7 @@ namespace CSVRender | ||||||
|             CSVWidget::SceneToolRun *mRun; |             CSVWidget::SceneToolRun *mRun; | ||||||
|             CSMDoc::Document& mDocument; |             CSMDoc::Document& mDocument; | ||||||
|             CSVWorld::PhysicsSystem *mPhysics; |             CSVWorld::PhysicsSystem *mPhysics; | ||||||
| 
 |             MouseState *mMouse; | ||||||
|             enum MouseState |  | ||||||
|             { |  | ||||||
|                 Mouse_Grab, |  | ||||||
|                 Mouse_Drag, |  | ||||||
|                 Mouse_Edit, |  | ||||||
|                 Mouse_Default |  | ||||||
|             }; |  | ||||||
|             MouseState mMouseState; |  | ||||||
| 
 |  | ||||||
|             QPoint mOldPos; |  | ||||||
|             std::string mCurrentObj; |  | ||||||
|             std::string mGrabbedSceneNode; |  | ||||||
|             QElapsedTimer *mMouseEventTimer; |  | ||||||
|             Ogre::Plane *mPlane; |  | ||||||
|             Ogre::Vector3 mOrigObjPos; |  | ||||||
|             Ogre::Vector3 mOrigMousePos; |  | ||||||
|             Ogre::Vector3 mCurrentMousePos; |  | ||||||
|             float mZOffset; |  | ||||||
|             std::map<std::string, std::vector<std::string> > mSelectedEntities; |  | ||||||
| 
 | 
 | ||||||
|         public: |         public: | ||||||
| 
 | 
 | ||||||
|  | @ -138,15 +119,6 @@ namespace CSVRender | ||||||
| 
 | 
 | ||||||
|             virtual std::string getStartupInstruction() = 0; |             virtual std::string getStartupInstruction() = 0; | ||||||
| 
 | 
 | ||||||
|             void placeObject(const std::string sceneNode, const Ogre::Vector3 &pos); |  | ||||||
|             std::pair<std::string, Ogre::Vector3> terrainUnderCursor(const int mouseX, const int mouseY); |  | ||||||
|             std::pair<std::string, Ogre::Vector3> objectUnderCursor(const int mouseX, const int mouseY); |  | ||||||
|             std::pair<bool, Ogre::Vector3> mousePositionOnPlane(const QPoint &pos, const Ogre::Plane &plane); |  | ||||||
|             void updateSelectionHighlight(const std::string sceneNode, const Ogre::Vector3 &position); |  | ||||||
| 
 |  | ||||||
|             void updateSceneWidgets(); |  | ||||||
|             bool isDebug(); |  | ||||||
| 
 |  | ||||||
|         private slots: |         private slots: | ||||||
| 
 | 
 | ||||||
|             void selectNavigationMode (const std::string& mode); |             void selectNavigationMode (const std::string& mode); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue