mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 22:26:37 +00:00 
			
		
		
		
	Merge remote-tracking branch 'glorf/rotate-update1'
This commit is contained in:
		
						commit
						1e9656ee9c
					
				
					 8 changed files with 154 additions and 12 deletions
				
			
		|  | @ -230,6 +230,8 @@ namespace MWBase | ||||||
| 
 | 
 | ||||||
|             virtual void rotateObject(const MWWorld::Ptr& ptr,float x,float y,float z, bool adjust = false) = 0; |             virtual void rotateObject(const MWWorld::Ptr& ptr,float x,float y,float z, bool adjust = false) = 0; | ||||||
| 
 | 
 | ||||||
|  |             virtual void localRotateObject (const MWWorld::Ptr& ptr, float x, float y, float z) = 0; | ||||||
|  | 
 | ||||||
|             virtual void safePlaceObject(const MWWorld::Ptr& ptr,MWWorld::CellStore &Cell,ESM::Position pos) = 0; |             virtual void safePlaceObject(const MWWorld::Ptr& ptr,MWWorld::CellStore &Cell,ESM::Position pos) = 0; | ||||||
|             ///< place an object in a "safe" location (ie not in the void, etc).
 |             ///< place an object in a "safe" location (ie not in the void, etc).
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -318,5 +318,9 @@ op 0x20001fb: DropSoulGem, explicit reference | ||||||
| op 0x20001fc: OnDeath | op 0x20001fc: OnDeath | ||||||
| op 0x20001fd: IsWerewolf | op 0x20001fd: IsWerewolf | ||||||
| op 0x20001fe: IsWerewolf, explicit reference | op 0x20001fe: IsWerewolf, explicit reference | ||||||
|  | op 0x20001ff: Rotate | ||||||
|  | op 0x2000200: Rotate, explicit reference | ||||||
|  | op 0x2000201: RotateWorld | ||||||
|  | op 0x2000202: RotateWorld, explicit reference | ||||||
| 
 | 
 | ||||||
| opcodes 0x20001ff-0x3ffffff unused | opcodes 0x2000203-0x3ffffff unused | ||||||
|  |  | ||||||
|  | @ -84,21 +84,27 @@ namespace MWScript | ||||||
|                     Interpreter::Type_Float angle = runtime[0].mFloat; |                     Interpreter::Type_Float angle = runtime[0].mFloat; | ||||||
|                     runtime.pop(); |                     runtime.pop(); | ||||||
| 
 | 
 | ||||||
|                     float ax = Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees(); |                     float ax = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[0]).valueDegrees(); | ||||||
|                     float ay = Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees(); |                     float ay = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[1]).valueDegrees(); | ||||||
|                     float az = Ogre::Radian(ptr.getRefData().getPosition().rot[2]).valueDegrees(); |                     float az = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[2]).valueDegrees(); | ||||||
|  | 
 | ||||||
|  |                     float *objRot = ptr.getRefData().getPosition().rot; | ||||||
|  | 
 | ||||||
|  |                     float lx = Ogre::Radian(objRot[0]).valueDegrees(); | ||||||
|  |                     float ly = Ogre::Radian(objRot[1]).valueDegrees(); | ||||||
|  |                     float lz = Ogre::Radian(objRot[2]).valueDegrees(); | ||||||
| 
 | 
 | ||||||
|                     if (axis == "x") |                     if (axis == "x") | ||||||
|                     { |                     { | ||||||
|                         MWBase::Environment::get().getWorld()->rotateObject(ptr,angle,ay,az); |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,angle-lx,ay,az); | ||||||
|                     } |                     } | ||||||
|                     else if (axis == "y") |                     else if (axis == "y") | ||||||
|                     { |                     { | ||||||
|                         MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,angle,az); |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,ax,angle-ly,az); | ||||||
|                     } |                     } | ||||||
|                     else if (axis == "z") |                     else if (axis == "z") | ||||||
|                     { |                     { | ||||||
|                         MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay,angle); |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,ax,ay,angle-lz); | ||||||
|                     } |                     } | ||||||
|                     else |                     else | ||||||
|                         throw std::runtime_error ("invalid ration axis: " + axis); |                         throw std::runtime_error ("invalid ration axis: " + axis); | ||||||
|  | @ -148,15 +154,15 @@ namespace MWScript | ||||||
| 
 | 
 | ||||||
|                     if (axis=="x") |                     if (axis=="x") | ||||||
|                     { |                     { | ||||||
|                         runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[0]).valueDegrees()); |                         runtime.push(Ogre::Radian(ptr.getCellRef().mPos.rot[0]).valueDegrees()+Ogre::Radian(ptr.getRefData().getLocalRotation().rot[0]).valueDegrees()); | ||||||
|                     } |                     } | ||||||
|                     else if (axis=="y") |                     else if (axis=="y") | ||||||
|                     { |                     { | ||||||
|                         runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[1]).valueDegrees()); |                         runtime.push(Ogre::Radian(ptr.getCellRef().mPos.rot[1]).valueDegrees()+Ogre::Radian(ptr.getRefData().getLocalRotation().rot[1]).valueDegrees()); | ||||||
|                     } |                     } | ||||||
|                     else if (axis=="z") |                     else if (axis=="z") | ||||||
|                     { |                     { | ||||||
|                         runtime.push(Ogre::Radian(ptr.getRefData().getPosition().rot[2]).valueDegrees()); |                         runtime.push(Ogre::Radian(ptr.getCellRef().mPos.rot[2]).valueDegrees()+Ogre::Radian(ptr.getRefData().getLocalRotation().rot[2]).valueDegrees()); | ||||||
|                     } |                     } | ||||||
|                     else |                     else | ||||||
|                         throw std::runtime_error ("invalid ration axis: " + axis); |                         throw std::runtime_error ("invalid ration axis: " + axis); | ||||||
|  | @ -542,6 +548,78 @@ namespace MWScript | ||||||
|                 } |                 } | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|  |         template<class R> | ||||||
|  |         class OpRotate : public Interpreter::Opcode0 | ||||||
|  |         { | ||||||
|  |             public: | ||||||
|  | 
 | ||||||
|  |                 virtual void execute (Interpreter::Runtime& runtime) | ||||||
|  |                 { | ||||||
|  |                     const MWWorld::Ptr& ptr = R()(runtime); | ||||||
|  | 
 | ||||||
|  |                     std::string axis = runtime.getStringLiteral (runtime[0].mInteger); | ||||||
|  |                     runtime.pop(); | ||||||
|  |                     Interpreter::Type_Float rotation = (runtime[0].mFloat*MWBase::Environment::get().getFrameDuration()); | ||||||
|  |                     runtime.pop(); | ||||||
|  | 
 | ||||||
|  |                     float ax = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[0]).valueDegrees(); | ||||||
|  |                     float ay = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[1]).valueDegrees(); | ||||||
|  |                     float az = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[2]).valueDegrees(); | ||||||
|  | 
 | ||||||
|  |                     if (axis == "x") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,ax+rotation,ay,az); | ||||||
|  |                     } | ||||||
|  |                     else if (axis == "y") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,ax,ay+rotation,az); | ||||||
|  |                     } | ||||||
|  |                     else if (axis == "z") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr,ax,ay,az+rotation); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                         throw std::runtime_error ("invalid ration axis: " + axis); | ||||||
|  |                 } | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         template<class R> | ||||||
|  |         class OpRotateWorld : public Interpreter::Opcode0 | ||||||
|  |         { | ||||||
|  |             public: | ||||||
|  | 
 | ||||||
|  |                 virtual void execute (Interpreter::Runtime& runtime) | ||||||
|  |                 { | ||||||
|  |                     MWWorld::Ptr ptr = R()(runtime); | ||||||
|  | 
 | ||||||
|  |                     std::string axis = runtime.getStringLiteral (runtime[0].mInteger); | ||||||
|  |                     runtime.pop(); | ||||||
|  |                     Interpreter::Type_Float rotation = (runtime[0].mFloat*MWBase::Environment::get().getFrameDuration()); | ||||||
|  |                     runtime.pop(); | ||||||
|  | 
 | ||||||
|  |                     float *objRot = ptr.getRefData().getPosition().rot; | ||||||
|  | 
 | ||||||
|  |                     float ax = Ogre::Radian(objRot[0]).valueDegrees(); | ||||||
|  |                     float ay = Ogre::Radian(objRot[1]).valueDegrees(); | ||||||
|  |                     float az = Ogre::Radian(objRot[2]).valueDegrees(); | ||||||
|  | 
 | ||||||
|  |                     if (axis == "x") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->rotateObject(ptr,ax+rotation,ay,az); | ||||||
|  |                     } | ||||||
|  |                     else if (axis == "y") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay+rotation,az); | ||||||
|  |                     } | ||||||
|  |                     else if (axis == "z") | ||||||
|  |                     { | ||||||
|  |                         MWBase::Environment::get().getWorld()->rotateObject(ptr,ax,ay,az+rotation); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                         throw std::runtime_error ("invalid rotation axis: " + axis); | ||||||
|  |                 } | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|         const int opcodeSetScale = 0x2000164; |         const int opcodeSetScale = 0x2000164; | ||||||
|         const int opcodeSetScaleExplicit = 0x2000165; |         const int opcodeSetScaleExplicit = 0x2000165; | ||||||
|         const int opcodeSetAngle = 0x2000166; |         const int opcodeSetAngle = 0x2000166; | ||||||
|  | @ -568,6 +646,10 @@ namespace MWScript | ||||||
|         const int opcodePlaceAtMeExplicit = 0x200019e; |         const int opcodePlaceAtMeExplicit = 0x200019e; | ||||||
|         const int opcodeModScale = 0x20001e3; |         const int opcodeModScale = 0x20001e3; | ||||||
|         const int opcodeModScaleExplicit = 0x20001e4; |         const int opcodeModScaleExplicit = 0x20001e4; | ||||||
|  |         const int opcodeRotate = 0x20001ff; | ||||||
|  |         const int opcodeRotateExplicit = 0x2000200; | ||||||
|  |         const int opcodeRotateWorld = 0x2000201; | ||||||
|  |         const int opcodeRotateWorldExplicit = 0x2000202; | ||||||
| 
 | 
 | ||||||
|         void registerExtensions (Compiler::Extensions& extensions) |         void registerExtensions (Compiler::Extensions& extensions) | ||||||
|         { |         { | ||||||
|  | @ -585,6 +667,8 @@ namespace MWScript | ||||||
|             extensions.registerInstruction("placeatpc","clfl",opcodePlaceAtPc); |             extensions.registerInstruction("placeatpc","clfl",opcodePlaceAtPc); | ||||||
|             extensions.registerInstruction("placeatme","clfl",opcodePlaceAtMe,opcodePlaceAtMeExplicit); |             extensions.registerInstruction("placeatme","clfl",opcodePlaceAtMe,opcodePlaceAtMeExplicit); | ||||||
|             extensions.registerInstruction("modscale","f",opcodeModScale,opcodeModScaleExplicit); |             extensions.registerInstruction("modscale","f",opcodeModScale,opcodeModScaleExplicit); | ||||||
|  |             extensions.registerInstruction("rotate","cf",opcodeRotate,opcodeRotateExplicit); | ||||||
|  |             extensions.registerInstruction("rotateworld","cf",opcodeRotateWorld,opcodeRotateWorldExplicit); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         void installOpcodes (Interpreter::Interpreter& interpreter) |         void installOpcodes (Interpreter::Interpreter& interpreter) | ||||||
|  | @ -614,6 +698,10 @@ namespace MWScript | ||||||
|             interpreter.installSegment5(opcodePlaceAtMeExplicit,new OpPlaceAtMe<ExplicitRef>); |             interpreter.installSegment5(opcodePlaceAtMeExplicit,new OpPlaceAtMe<ExplicitRef>); | ||||||
|             interpreter.installSegment5(opcodeModScale,new OpModScale<ImplicitRef>); |             interpreter.installSegment5(opcodeModScale,new OpModScale<ImplicitRef>); | ||||||
|             interpreter.installSegment5(opcodeModScaleExplicit,new OpModScale<ExplicitRef>); |             interpreter.installSegment5(opcodeModScaleExplicit,new OpModScale<ExplicitRef>); | ||||||
|  |             interpreter.installSegment5(opcodeRotate,new OpRotate<ImplicitRef>); | ||||||
|  |             interpreter.installSegment5(opcodeRotateExplicit,new OpRotate<ExplicitRef>); | ||||||
|  |             interpreter.installSegment5(opcodeRotateWorld,new OpRotateWorld<ImplicitRef>); | ||||||
|  |             interpreter.installSegment5(opcodeRotateWorldExplicit,new OpRotateWorld<ExplicitRef>); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -19,6 +19,7 @@ namespace MWWorld | ||||||
|         mEnabled = refData.mEnabled; |         mEnabled = refData.mEnabled; | ||||||
|         mCount = refData.mCount; |         mCount = refData.mCount; | ||||||
|         mPosition = refData.mPosition; |         mPosition = refData.mPosition; | ||||||
|  |         mLocalRotation = refData.mLocalRotation; | ||||||
| 
 | 
 | ||||||
|         mCustomData = refData.mCustomData ? refData.mCustomData->clone() : 0; |         mCustomData = refData.mCustomData ? refData.mCustomData->clone() : 0; | ||||||
|     } |     } | ||||||
|  | @ -34,7 +35,11 @@ namespace MWWorld | ||||||
|     RefData::RefData (const ESM::CellRef& cellRef) |     RefData::RefData (const ESM::CellRef& cellRef) | ||||||
|     : mBaseNode(0), mHasLocals (false), mEnabled (true), mCount (1), mPosition (cellRef.mPos), |     : mBaseNode(0), mHasLocals (false), mEnabled (true), mCount (1), mPosition (cellRef.mPos), | ||||||
|       mCustomData (0) |       mCustomData (0) | ||||||
|     {} |     { | ||||||
|  |         mLocalRotation.rot[0]=0; | ||||||
|  |         mLocalRotation.rot[1]=0; | ||||||
|  |         mLocalRotation.rot[2]=0; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     RefData::RefData (const RefData& refData) |     RefData::RefData (const RefData& refData) | ||||||
|     : mBaseNode(0), mCustomData (0) |     : mBaseNode(0), mCustomData (0) | ||||||
|  | @ -141,6 +146,11 @@ namespace MWWorld | ||||||
|         return mPosition; |         return mPosition; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     LocalRotation& RefData::getLocalRotation() | ||||||
|  |     { | ||||||
|  |         return mLocalRotation; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void RefData::setCustomData (CustomData *data) |     void RefData::setCustomData (CustomData *data) | ||||||
|     { |     { | ||||||
|         delete mCustomData; |         delete mCustomData; | ||||||
|  |  | ||||||
|  | @ -18,6 +18,10 @@ namespace ESM | ||||||
| 
 | 
 | ||||||
| namespace MWWorld | namespace MWWorld | ||||||
| { | { | ||||||
|  |     struct LocalRotation{ | ||||||
|  |         float rot[3]; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     class CustomData; |     class CustomData; | ||||||
| 
 | 
 | ||||||
|     class RefData |     class RefData | ||||||
|  | @ -34,6 +38,8 @@ namespace MWWorld | ||||||
| 
 | 
 | ||||||
|             ESM::Position mPosition; |             ESM::Position mPosition; | ||||||
| 
 | 
 | ||||||
|  |             LocalRotation mLocalRotation; | ||||||
|  | 
 | ||||||
|             CustomData *mCustomData; |             CustomData *mCustomData; | ||||||
| 
 | 
 | ||||||
|             void copy (const RefData& refData); |             void copy (const RefData& refData); | ||||||
|  | @ -78,6 +84,8 @@ namespace MWWorld | ||||||
| 
 | 
 | ||||||
|             ESM::Position& getPosition(); |             ESM::Position& getPosition(); | ||||||
| 
 | 
 | ||||||
|  |             LocalRotation& getLocalRotation(); | ||||||
|  | 
 | ||||||
|             void setCustomData (CustomData *data); |             void setCustomData (CustomData *data); | ||||||
|             ///< Set custom data (potentially replacing old custom data). The ownership of \æ data is
 |             ///< Set custom data (potentially replacing old custom data). The ownership of \æ data is
 | ||||||
|             /// transferred to this.
 |             /// transferred to this.
 | ||||||
|  |  | ||||||
|  | @ -49,7 +49,12 @@ namespace | ||||||
|                     { |                     { | ||||||
|                         rendering.addObject(ptr); |                         rendering.addObject(ptr); | ||||||
|                         class_.insertObject(ptr, physics); |                         class_.insertObject(ptr, physics); | ||||||
|                         MWBase::Environment::get().getWorld()->rotateObject(ptr, 0, 0, 0, true); | 
 | ||||||
|  |                         float ax = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[0]).valueDegrees(); | ||||||
|  |                         float ay = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[1]).valueDegrees(); | ||||||
|  |                         float az = Ogre::Radian(ptr.getRefData().getLocalRotation().rot[2]).valueDegrees(); | ||||||
|  |                         MWBase::Environment::get().getWorld()->localRotateObject(ptr, ax, ay, az); | ||||||
|  | 
 | ||||||
|                         MWBase::Environment::get().getWorld()->scaleObject(ptr, ptr.getCellRef().mScale); |                         MWBase::Environment::get().getWorld()->scaleObject(ptr, ptr.getCellRef().mScale); | ||||||
|                         class_.adjustPosition(ptr); |                         class_.adjustPosition(ptr); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|  | @ -824,6 +824,29 @@ namespace MWWorld | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     void World::localRotateObject (const Ptr& ptr, float x, float y, float z) | ||||||
|  |     { | ||||||
|  |         if (ptr.getRefData().getBaseNode() != 0) { | ||||||
|  | 
 | ||||||
|  |             ptr.getRefData().getLocalRotation().rot[0]=Ogre::Degree(x).valueRadians(); | ||||||
|  |             ptr.getRefData().getLocalRotation().rot[1]=Ogre::Degree(y).valueRadians(); | ||||||
|  |             ptr.getRefData().getLocalRotation().rot[2]=Ogre::Degree(z).valueRadians(); | ||||||
|  | 
 | ||||||
|  |             float *worldRot = ptr.getRefData().getPosition().rot; | ||||||
|  | 
 | ||||||
|  |             Ogre::Quaternion worldRotQuat(Ogre::Quaternion(Ogre::Radian(-worldRot[0]), Ogre::Vector3::UNIT_X)* | ||||||
|  |             Ogre::Quaternion(Ogre::Radian(-worldRot[1]), Ogre::Vector3::UNIT_Y)* | ||||||
|  |             Ogre::Quaternion(Ogre::Radian(-worldRot[2]), Ogre::Vector3::UNIT_Z)); | ||||||
|  | 
 | ||||||
|  |             Ogre::Quaternion rot(Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-x).valueRadians()), Ogre::Vector3::UNIT_X)* | ||||||
|  |             Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-y).valueRadians()), Ogre::Vector3::UNIT_Y)* | ||||||
|  |             Ogre::Quaternion(Ogre::Radian(Ogre::Degree(-z).valueRadians()), Ogre::Vector3::UNIT_Z)); | ||||||
|  | 
 | ||||||
|  |             ptr.getRefData().getBaseNode()->setOrientation(worldRotQuat*rot); | ||||||
|  |             mPhysics->rotateObject(ptr); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void World::adjustPosition(const Ptr &ptr) |     void World::adjustPosition(const Ptr &ptr) | ||||||
|     { |     { | ||||||
|         Ogre::Vector3 pos (ptr.getRefData().getPosition().pos[0], ptr.getRefData().getPosition().pos[1], ptr.getRefData().getPosition().pos[2]); |         Ogre::Vector3 pos (ptr.getRefData().getPosition().pos[0], ptr.getRefData().getPosition().pos[1], ptr.getRefData().getPosition().pos[2]); | ||||||
|  |  | ||||||
|  | @ -254,6 +254,8 @@ namespace MWWorld | ||||||
|             /// \param adjust indicates rotation should be set or adjusted
 |             /// \param adjust indicates rotation should be set or adjusted
 | ||||||
|             virtual void rotateObject (const Ptr& ptr,float x,float y,float z, bool adjust = false); |             virtual void rotateObject (const Ptr& ptr,float x,float y,float z, bool adjust = false); | ||||||
| 
 | 
 | ||||||
|  |             virtual void localRotateObject (const Ptr& ptr, float x, float y, float z); | ||||||
|  | 
 | ||||||
|             virtual void safePlaceObject(const MWWorld::Ptr& ptr,MWWorld::CellStore &Cell,ESM::Position pos); |             virtual void safePlaceObject(const MWWorld::Ptr& ptr,MWWorld::CellStore &Cell,ESM::Position pos); | ||||||
|             ///< place an object in a "safe" location (ie not in the void, etc). Makes a copy of the Ptr.
 |             ///< place an object in a "safe" location (ie not in the void, etc). Makes a copy of the Ptr.
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue