1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-26 20:56:37 +00:00

Implement HitOnMe script function

This commit is contained in:
Chris Robinson 2013-07-26 08:08:52 -07:00
parent 5379e607cb
commit 3298eb1b37
9 changed files with 69 additions and 15 deletions

View file

@ -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")
{ {

View file

@ -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")
{ {

View file

@ -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);

View file

@ -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

View file

@ -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}");
} }
} }

View file

@ -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;
}
} }

View file

@ -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;
}; };
} }

View file

@ -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

View file

@ -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>);
} }
} }
} }