mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 19:26:37 +00:00 
			
		
		
		
	fix for pickup up transparent objects, workaround for doors
This commit is contained in:
		
							parent
							
								
									30407d3a2c
								
							
						
					
					
						commit
						80456778be
					
				
					 3 changed files with 85 additions and 11 deletions
				
			
		|  | @ -5,6 +5,7 @@ | |||
| #include <OgreBillboardSet.h> | ||||
| #include <OgreHardwareOcclusionQuery.h> | ||||
| #include <OgreEntity.h> | ||||
| #include <OgreSubEntity.h> | ||||
| 
 | ||||
| using namespace MWRender; | ||||
| using namespace Ogre; | ||||
|  | @ -12,7 +13,7 @@ using namespace Ogre; | |||
| OcclusionQuery::OcclusionQuery(OEngine::Render::OgreRenderer* renderer, SceneNode* sunNode) : | ||||
|     mSunTotalAreaQuery(0), mSunVisibleAreaQuery(0), mSingleObjectQuery(0), mActiveQuery(0), | ||||
|     mDoQuery(0), mSunVisibility(0), mQuerySingleObjectStarted(false), mTestResult(false),  | ||||
|     mQuerySingleObjectRequested(false), mWasVisible(false), mObjectWasVisible(false), mDoQuery2(false) | ||||
|     mQuerySingleObjectRequested(false), mWasVisible(false), mObjectWasVisible(false) | ||||
| { | ||||
|     mRendering = renderer; | ||||
|     mSunNode = sunNode; | ||||
|  | @ -82,7 +83,6 @@ OcclusionQuery::OcclusionQuery(OEngine::Render::OgreRenderer* renderer, SceneNod | |||
|     mRendering->getScene()->addRenderObjectListener(this); | ||||
|     mRendering->getScene()->addRenderQueueListener(this); | ||||
|     mDoQuery = true; | ||||
|     mDoQuery2 = true; | ||||
| } | ||||
| 
 | ||||
| OcclusionQuery::~OcclusionQuery() | ||||
|  | @ -190,7 +190,6 @@ void OcclusionQuery::update(float duration) | |||
|     // Stop occlusion queries until we get their information
 | ||||
|     // (may not happen on the same frame they are requested in)
 | ||||
|     mDoQuery = false; | ||||
|     mDoQuery2 = false; | ||||
| 
 | ||||
|     if (!mSunTotalAreaQuery->isStillOutstanding() | ||||
|         && !mSunVisibleAreaQuery->isStillOutstanding() | ||||
|  | @ -252,3 +251,40 @@ bool OcclusionQuery::getTestResult() | |||
| 
 | ||||
|     return mTestResult; | ||||
| } | ||||
| 
 | ||||
| bool OcclusionQuery::isPotentialOccluder(Ogre::SceneNode* node) | ||||
| { | ||||
|     bool result = false; | ||||
|     for (unsigned int i=0; i < node->numAttachedObjects(); ++i) | ||||
|     { | ||||
|         MovableObject* ob = node->getAttachedObject(i); | ||||
|         std::string type = ob->getMovableType(); | ||||
|         if (type == "Entity") | ||||
|         { | ||||
|             Entity* ent = static_cast<Entity*>(ob); | ||||
|             for (unsigned int j=0; j < ent->getNumSubEntities(); ++j) | ||||
|             { | ||||
|                 // if any sub entity has a material with depth write off,
 | ||||
|                 // consider the object as not an occluder
 | ||||
|                 MaterialPtr mat = ent->getSubEntity(j)->getMaterial(); | ||||
| 
 | ||||
|                 Material::TechniqueIterator techIt = mat->getTechniqueIterator(); | ||||
|                 while (techIt.hasMoreElements()) | ||||
|                 { | ||||
|                     Technique* tech = techIt.getNext(); | ||||
|                     Technique::PassIterator passIt = tech->getPassIterator(); | ||||
|                     while (passIt.hasMoreElements()) | ||||
|                     { | ||||
|                         Pass* pass = passIt.getNext(); | ||||
| 
 | ||||
|                         if (pass->getDepthWriteEnabled() == false) | ||||
|                             return false; | ||||
|                         else | ||||
|                             result = true; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     return result; | ||||
| } | ||||
|  |  | |||
|  | @ -46,6 +46,14 @@ namespace MWRender | |||
|          */ | ||||
|         bool occlusionTestPending(); | ||||
| 
 | ||||
|         /**
 | ||||
|          * Checks if the objects held by this scenenode | ||||
|          * can be considered as potential occluders | ||||
|          * (which might not be the case when transparency is involved) | ||||
|          * @param Scene node | ||||
|          */ | ||||
|         bool isPotentialOccluder(Ogre::SceneNode* node); | ||||
| 
 | ||||
|         /**
 | ||||
|          * @return true if the object tested in the last request was occluded | ||||
|          */ | ||||
|  |  | |||
|  | @ -750,15 +750,16 @@ namespace MWWorld | |||
|                 // figure out which object we want to test against
 | ||||
|                 std::vector < std::pair < float, std::string > > results = mPhysics->getFacedObjects(); | ||||
| 
 | ||||
|                 // ignore the player
 | ||||
|                 for (std::vector < std::pair < float, std::string > >::iterator it = results.begin(); | ||||
|                     it != results.end(); ++it) | ||||
|                 // ignore the player and other things we're not interested in
 | ||||
|                 std::vector < std::pair < float, std::string > >::iterator it = results.begin(); | ||||
|                 while (it != results.end()) | ||||
|                 { | ||||
|                     if ( (*it).second == mPlayer->getPlayer().getRefData().getHandle() ) | ||||
|                     if ( (*it).second == mPlayer->getPlayer().getRefData().getHandle()) | ||||
|                     { | ||||
|                         results.erase(it); | ||||
|                         break; | ||||
|                         it = results.erase(it); | ||||
|                     } | ||||
|                     else | ||||
|                         ++it; | ||||
|                 } | ||||
| 
 | ||||
|                 if (results.size() == 0) | ||||
|  | @ -774,6 +775,7 @@ namespace MWWorld | |||
|                     btVector3 p = mPhysics->getRayPoint(results.front().first); | ||||
|                     Ogre::Vector3 pos(p.x(), p.z(), -p.y()); | ||||
|                     Ogre::SceneNode* node = mFaced1.getRefData().getBaseNode(); | ||||
| 
 | ||||
|                     query->occlusionTest(pos, node); | ||||
|                 } | ||||
|                 else | ||||
|  | @ -784,10 +786,38 @@ namespace MWWorld | |||
|                     mFaced2 = getPtrViaHandle(results[1].second); | ||||
|                     mNumFacing = 2; | ||||
| 
 | ||||
|                     //std::cout << "Num facing 2 : " << mFaced1Name << " " << mFaced2Name << std::endl;
 | ||||
|                     //std::cout << "Type 1 " << mFaced1.getTypeName() << " " << mFaced2.getTypeName() << std::endl;
 | ||||
| 
 | ||||
|                     btVector3 p = mPhysics->getRayPoint(results[1].first); | ||||
|                     Ogre::Vector3 pos(p.x(), p.z(), -p.y()); | ||||
|                     Ogre::SceneNode* node = mFaced2.getRefData().getBaseNode(); | ||||
|                     query->occlusionTest(pos, node); | ||||
|                     Ogre::SceneNode* node1 = mFaced1.getRefData().getBaseNode(); | ||||
|                     Ogre::SceneNode* node2 = mFaced2.getRefData().getBaseNode(); | ||||
| 
 | ||||
|                     // no need to test if the first node is not occluder
 | ||||
|                     if (!query->isPotentialOccluder(node1) && (mFaced1.getTypeName().find("Static") == std::string::npos)) | ||||
|                     { | ||||
|                         mFacedHandle = mFaced1Name; | ||||
|                         //std::cout << "node1 Not an occluder" << std::endl;
 | ||||
|                         return; | ||||
|                     } | ||||
| 
 | ||||
|                     // no need to test if the second object is static (thus cannot be activated)
 | ||||
|                     if (mFaced2.getTypeName().find("Static") != std::string::npos) | ||||
|                     { | ||||
|                         mFacedHandle = mFaced1Name; | ||||
|                         return; | ||||
|                     } | ||||
| 
 | ||||
|                     // work around door problems
 | ||||
|                     if (mFaced1.getTypeName().find("Static") != std::string::npos | ||||
|                         && mFaced2.getTypeName().find("Door") != std::string::npos) | ||||
|                     { | ||||
|                         mFacedHandle = mFaced2Name; | ||||
|                         return; | ||||
|                     } | ||||
| 
 | ||||
|                     query->occlusionTest(pos, node2); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue