mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 18:26:41 +00:00 
			
		
		
		
	Implement HitOnMe script function
This commit is contained in:
		
							parent
							
								
									5379e607cb
								
							
						
					
					
						commit
						3298eb1b37
					
				
					 9 changed files with 69 additions and 15 deletions
				
			
		|  | @ -167,7 +167,8 @@ namespace MWClass | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // TODO: Handle HitOnMe script function
 |         if(!object.isEmpty()) | ||||||
|  |             getCreatureStats(ptr).setLastHitObject(MWWorld::Class::get(object).getId(object)); | ||||||
| 
 | 
 | ||||||
|         if(!attacker.isEmpty() && attacker.getRefData().getHandle() == "player") |         if(!attacker.isEmpty() && attacker.getRefData().getHandle() == "player") | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -392,7 +392,8 @@ namespace MWClass | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // TODO: Handle HitOnMe script function
 |         if(!object.isEmpty()) | ||||||
|  |             getCreatureStats(ptr).setLastHitObject(MWWorld::Class::get(object).getId(object)); | ||||||
| 
 | 
 | ||||||
|         if(!attacker.isEmpty() && attacker.getRefData().getHandle() == "player") |         if(!attacker.isEmpty() && attacker.getRefData().getHandle() == "player") | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -22,6 +22,13 @@ | ||||||
| 
 | 
 | ||||||
| namespace MWClass | namespace MWClass | ||||||
| { | { | ||||||
|  |     std::string Weapon::getId (const MWWorld::Ptr& ptr) const | ||||||
|  |     { | ||||||
|  |         MWWorld::LiveCellRef<ESM::Weapon> *ref = ptr.get<ESM::Weapon>(); | ||||||
|  | 
 | ||||||
|  |         return ref->mBase->mId; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void Weapon::insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const |     void Weapon::insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const | ||||||
|     { |     { | ||||||
|         const std::string model = getModel(ptr); |         const std::string model = getModel(ptr); | ||||||
|  |  | ||||||
|  | @ -12,6 +12,9 @@ namespace MWClass | ||||||
| 
 | 
 | ||||||
|         public: |         public: | ||||||
| 
 | 
 | ||||||
|  |             virtual std::string getId (const MWWorld::Ptr& ptr) const; | ||||||
|  |             ///< Return ID of \a ptr
 | ||||||
|  | 
 | ||||||
|             virtual void insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const; |             virtual void insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const; | ||||||
|             ///< Add reference into a cell for rendering
 |             ///< Add reference into a cell for rendering
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -221,7 +221,11 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|             for(PtrControllerMap::iterator iter(mActors.begin());iter != mActors.end();iter++) |             for(PtrControllerMap::iterator iter(mActors.begin());iter != mActors.end();iter++) | ||||||
|             { |             { | ||||||
|                 if(!MWWorld::Class::get(iter->first).getCreatureStats(iter->first).isDead()) |                 const MWWorld::Class &cls = MWWorld::Class::get(iter->first); | ||||||
|  |                 CreatureStats &stats = cls.getCreatureStats(iter->first); | ||||||
|  | 
 | ||||||
|  |                 stats.setLastHitObject(std::string()); | ||||||
|  |                 if(!stats.isDead()) | ||||||
|                 { |                 { | ||||||
|                     if(iter->second->isDead()) |                     if(iter->second->isDead()) | ||||||
|                         iter->second->resurrect(); |                         iter->second->resurrect(); | ||||||
|  | @ -230,7 +234,7 @@ namespace MWMechanics | ||||||
|                     if(iter->first.getTypeName() == typeid(ESM::NPC).name()) |                     if(iter->first.getTypeName() == typeid(ESM::NPC).name()) | ||||||
|                         updateNpc(iter->first, totalDuration, paused); |                         updateNpc(iter->first, totalDuration, paused); | ||||||
| 
 | 
 | ||||||
|                     if(!MWWorld::Class::get(iter->first).getCreatureStats(iter->first).isDead()) |                     if(!stats.isDead()) | ||||||
|                         continue; |                         continue; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|  | @ -238,16 +242,15 @@ namespace MWMechanics | ||||||
|                 // \todo remove workaround, once player death can be handled
 |                 // \todo remove workaround, once player death can be handled
 | ||||||
|                 if(iter->first.getRefData().getHandle()=="player") |                 if(iter->first.getRefData().getHandle()=="player") | ||||||
|                 { |                 { | ||||||
|                     MWMechanics::DynamicStat<float> stat ( |                     MWMechanics::DynamicStat<float> stat(stats.getHealth()); | ||||||
|                         MWWorld::Class::get(iter->first).getCreatureStats(iter->first).getHealth()); |  | ||||||
| 
 | 
 | ||||||
|                     if (stat.getModified()<1) |                     if(stat.getModified()<1) | ||||||
|                     { |                     { | ||||||
|                         stat.setModified (1, 0); |                         stat.setModified(1, 0); | ||||||
|                         MWWorld::Class::get(iter->first).getCreatureStats(iter->first).setHealth(stat); |                         stats.setHealth(stat); | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                     MWWorld::Class::get(iter->first).getCreatureStats(iter->first).resurrect(); |                     stats.resurrect(); | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|  | @ -256,11 +259,10 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|                 iter->second->kill(); |                 iter->second->kill(); | ||||||
| 
 | 
 | ||||||
|                 ++mDeathCount[MWWorld::Class::get(iter->first).getId(iter->first)]; |                 ++mDeathCount[cls.getId(iter->first)]; | ||||||
| 
 | 
 | ||||||
|                 if(MWWorld::Class::get(iter->first).isEssential(iter->first)) |                 if(cls.isEssential(iter->first)) | ||||||
|                     MWBase::Environment::get().getWindowManager()->messageBox( |                     MWBase::Environment::get().getWindowManager()->messageBox("#{sKilledEssential}"); | ||||||
|                         "#{sKilledEssential}"); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -318,4 +318,14 @@ namespace MWMechanics | ||||||
|     { |     { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     void CreatureStats::setLastHitObject(const std::string& objectid) | ||||||
|  |     { | ||||||
|  |         mLastHitObject = objectid; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const std::string &CreatureStats::getLastHitObject() const | ||||||
|  |     { | ||||||
|  |         return mLastHitObject; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -38,6 +38,8 @@ namespace MWMechanics | ||||||
| 
 | 
 | ||||||
|         int mAttackType; |         int mAttackType; | ||||||
| 
 | 
 | ||||||
|  |         std::string mLastHitObject; // The last object to hit this actor
 | ||||||
|  | 
 | ||||||
|     public: |     public: | ||||||
|         CreatureStats(); |         CreatureStats(); | ||||||
| 
 | 
 | ||||||
|  | @ -155,6 +157,9 @@ namespace MWMechanics | ||||||
|         void setHostile (bool hostile); |         void setHostile (bool hostile); | ||||||
| 
 | 
 | ||||||
|         bool getCreatureTargetted() const; |         bool getCreatureTargetted() const; | ||||||
|  | 
 | ||||||
|  |         void setLastHitObject(const std::string &objectid); | ||||||
|  |         const std::string &getLastHitObject() const; | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -338,5 +338,7 @@ op 0x200020f: GetStandingActor, explicit | ||||||
| op 0x2000210: GetStartingAngle | op 0x2000210: GetStartingAngle | ||||||
| op 0x2000211: GetStartingAngle, explicit | op 0x2000211: GetStartingAngle, explicit | ||||||
| op 0x2000212: GetWindSpeed | op 0x2000212: GetWindSpeed | ||||||
|  | op 0x2000213: HitOnMe | ||||||
|  | op 0x2000214: HitOnMe, explicit | ||||||
| 
 | 
 | ||||||
| opcodes 0x2000213-0x3ffffff unused | opcodes 0x2000215-0x3ffffff unused | ||||||
|  |  | ||||||
|  | @ -599,6 +599,23 @@ namespace MWScript | ||||||
|                 } |                 } | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|  |         template <class R> | ||||||
|  |         class OpHitOnMe : public Interpreter::Opcode0 | ||||||
|  |         { | ||||||
|  |             public: | ||||||
|  | 
 | ||||||
|  |                 virtual void execute (Interpreter::Runtime& runtime) | ||||||
|  |                 { | ||||||
|  |                     MWWorld::Ptr ptr = R()(runtime); | ||||||
|  | 
 | ||||||
|  |                     std::string objectID = runtime.getStringLiteral (runtime[0].mInteger); | ||||||
|  |                     runtime.pop(); | ||||||
|  | 
 | ||||||
|  |                     MWMechanics::CreatureStats &stats = MWWorld::Class::get(ptr).getCreatureStats(ptr); | ||||||
|  |                     runtime.push(::Misc::StringUtils::ciEqual(objectID, stats.getLastHitObject())); | ||||||
|  |                 } | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|         const int opcodeXBox = 0x200000c; |         const int opcodeXBox = 0x200000c; | ||||||
|         const int opcodeOnActivate = 0x200000d; |         const int opcodeOnActivate = 0x200000d; | ||||||
|         const int opcodeActivate = 0x2000075; |         const int opcodeActivate = 0x2000075; | ||||||
|  | @ -650,6 +667,9 @@ namespace MWScript | ||||||
| 
 | 
 | ||||||
|         const int opcodePlayBink = 0x20001f7; |         const int opcodePlayBink = 0x20001f7; | ||||||
| 
 | 
 | ||||||
|  |         const int opcodeHitOnMe = 0x2000213; | ||||||
|  |         const int opcodeHitOnMeExplicit = 0x2000214; | ||||||
|  | 
 | ||||||
|         void registerExtensions (Compiler::Extensions& extensions) |         void registerExtensions (Compiler::Extensions& extensions) | ||||||
|         { |         { | ||||||
|             extensions.registerFunction ("xbox", 'l', "", opcodeXBox); |             extensions.registerFunction ("xbox", 'l', "", opcodeXBox); | ||||||
|  | @ -692,6 +712,7 @@ namespace MWScript | ||||||
|             extensions.registerFunction ("getstandingpc", 'l', "", opcodeGetStandingPc, opcodeGetStandingPcExplicit); |             extensions.registerFunction ("getstandingpc", 'l', "", opcodeGetStandingPc, opcodeGetStandingPcExplicit); | ||||||
|             extensions.registerFunction ("getstandingactor", 'l', "", opcodeGetStandingActor, opcodeGetStandingActorExplicit); |             extensions.registerFunction ("getstandingactor", 'l', "", opcodeGetStandingActor, opcodeGetStandingActorExplicit); | ||||||
|             extensions.registerFunction ("getwindspeed", 'f', "", opcodeGetWindSpeed); |             extensions.registerFunction ("getwindspeed", 'f', "", opcodeGetWindSpeed); | ||||||
|  |             extensions.registerFunction ("hitonme", 'l', "S", opcodeHitOnMe, opcodeHitOnMeExplicit); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         void installOpcodes (Interpreter::Interpreter& interpreter) |         void installOpcodes (Interpreter::Interpreter& interpreter) | ||||||
|  | @ -745,6 +766,8 @@ namespace MWScript | ||||||
|             interpreter.installSegment5 (opcodeGetStandingActor, new OpGetStandingActor<ImplicitRef>); |             interpreter.installSegment5 (opcodeGetStandingActor, new OpGetStandingActor<ImplicitRef>); | ||||||
|             interpreter.installSegment5 (opcodeGetStandingActorExplicit, new OpGetStandingActor<ExplicitRef>); |             interpreter.installSegment5 (opcodeGetStandingActorExplicit, new OpGetStandingActor<ExplicitRef>); | ||||||
|             interpreter.installSegment5 (opcodeGetWindSpeed, new OpGetWindSpeed); |             interpreter.installSegment5 (opcodeGetWindSpeed, new OpGetWindSpeed); | ||||||
|  |             interpreter.installSegment5 (opcodeHitOnMe, new OpHitOnMe<ImplicitRef>); | ||||||
|  |             interpreter.installSegment5 (opcodeHitOnMeExplicit, new OpHitOnMe<ExplicitRef>); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue