|
|
@ -14,6 +14,7 @@
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwmechanics/aiactivate.hpp"
|
|
|
|
#include "../mwmechanics/aiactivate.hpp"
|
|
|
|
#include "../mwmechanics/aiescort.hpp"
|
|
|
|
#include "../mwmechanics/aiescort.hpp"
|
|
|
@ -50,6 +51,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiActivate activatePackage(objectID);
|
|
|
|
MWMechanics::AiActivate activatePackage(objectID);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(activatePackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(activatePackage, ptr);
|
|
|
|
Log(Debug::Info) << "AiActivate";
|
|
|
|
Log(Debug::Info) << "AiActivate";
|
|
|
@ -77,6 +81,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiTravel travelPackage(x, y, z);
|
|
|
|
MWMechanics::AiTravel travelPackage(x, y, z);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(travelPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(travelPackage, ptr);
|
|
|
|
|
|
|
|
|
|
|
@ -111,6 +118,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiEscort escortPackage(actorID, static_cast<int>(duration), x, y, z);
|
|
|
|
MWMechanics::AiEscort escortPackage(actorID, static_cast<int>(duration), x, y, z);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
|
|
|
|
|
|
|
|
|
|
@ -148,10 +158,14 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
if (cellID.empty())
|
|
|
|
if (cellID.empty())
|
|
|
|
throw std::runtime_error("AiEscortCell: no cell ID given");
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Cell>().find(cellID);
|
|
|
|
if (!MWBase::Environment::get().getWorld()->getStore().get<ESM::Cell>().search(cellID))
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiEscort escortPackage(actorID, cellID, static_cast<int>(duration), x, y, z);
|
|
|
|
MWMechanics::AiEscort escortPackage(actorID, cellID, static_cast<int>(duration), x, y, z);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
|
|
@ -169,9 +183,11 @@ namespace MWScript
|
|
|
|
{
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
|
|
|
|
Interpreter::Type_Integer value = ptr.getClass().getCreatureStats (ptr).getAiSequence().isPackageDone();
|
|
|
|
bool done = false;
|
|
|
|
|
|
|
|
if (ptr.getClass().isActor())
|
|
|
|
|
|
|
|
done = ptr.getClass().getCreatureStats(ptr).getAiSequence().isPackageDone();
|
|
|
|
|
|
|
|
|
|
|
|
runtime.push (value);
|
|
|
|
runtime.push(done);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
@ -224,6 +240,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiWander wanderPackage(range, duration, time, idleList, repeat);
|
|
|
|
MWMechanics::AiWander wanderPackage(range, duration, time, idleList, repeat);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(wanderPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(wanderPackage, ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -240,7 +259,10 @@ namespace MWScript
|
|
|
|
{
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
|
|
|
|
runtime.push(ptr.getClass().getCreatureStats (ptr).getAiSetting (mIndex).getModified(false));
|
|
|
|
Interpreter::Type_Integer value = 0;
|
|
|
|
|
|
|
|
if (ptr.getClass().isActor())
|
|
|
|
|
|
|
|
value = ptr.getClass().getCreatureStats (ptr).getAiSetting(mIndex).getModified(false);
|
|
|
|
|
|
|
|
runtime.push(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
template<class R>
|
|
|
|
template<class R>
|
|
|
@ -256,6 +278,9 @@ namespace MWScript
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
|
|
runtime.pop();
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
int modified = ptr.getClass().getCreatureStats (ptr).getAiSetting (mIndex).getBase() + value;
|
|
|
|
int modified = ptr.getClass().getCreatureStats (ptr).getAiSetting (mIndex).getBase() + value;
|
|
|
|
|
|
|
|
|
|
|
|
ptr.getClass().getCreatureStats (ptr).setAiSetting (mIndex, modified);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).setAiSetting (mIndex, modified);
|
|
|
@ -309,6 +334,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiFollow followPackage(actorID, duration, x, y ,z);
|
|
|
|
MWMechanics::AiFollow followPackage(actorID, duration, x, y ,z);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
|
|
|
|
|
|
|
|
|
|
@ -346,6 +374,9 @@ namespace MWScript
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!ptr.getClass().isActor() || ptr == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiFollow followPackage(actorID, cellID, duration, x, y ,z);
|
|
|
|
MWMechanics::AiFollow followPackage(actorID, cellID, duration, x, y ,z);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
|
|
|
Log(Debug::Info) << "AiFollow: " << actorID << ", " << x << ", " << y << ", " << z << ", " << duration;
|
|
|
|
Log(Debug::Info) << "AiFollow: " << actorID << ", " << x << ", " << y << ", " << z << ", " << duration;
|
|
|
@ -431,11 +462,12 @@ namespace MWScript
|
|
|
|
std::string testedTargetId = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
std::string testedTargetId = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
const MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool targetsAreEqual = false;
|
|
|
|
bool targetsAreEqual = false;
|
|
|
|
|
|
|
|
if (actor.getClass().isActor())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
const MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
|
|
|
MWWorld::Ptr targetPtr;
|
|
|
|
MWWorld::Ptr targetPtr;
|
|
|
|
if (creatureStats.getAiSequence().getCombatTarget (targetPtr))
|
|
|
|
if (creatureStats.getAiSequence().getCombatTarget(targetPtr))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!targetPtr.isEmpty() && targetPtr.getCellRef().getRefId() == testedTargetId)
|
|
|
|
if (!targetPtr.isEmpty() && targetPtr.getCellRef().getRefId() == testedTargetId)
|
|
|
|
targetsAreEqual = true;
|
|
|
|
targetsAreEqual = true;
|
|
|
@ -447,7 +479,8 @@ namespace MWScript
|
|
|
|
bool sayActive = MWBase::Environment::get().getSoundManager()->sayActive(actor);
|
|
|
|
bool sayActive = MWBase::Environment::get().getSoundManager()->sayActive(actor);
|
|
|
|
targetsAreEqual = (greeting && sayActive) || mechMgr->isTurningToPlayer(actor);
|
|
|
|
targetsAreEqual = (greeting && sayActive) || mechMgr->isTurningToPlayer(actor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
runtime.push(int(targetsAreEqual));
|
|
|
|
}
|
|
|
|
|
|
|
|
runtime.push(targetsAreEqual);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
@ -474,6 +507,8 @@ namespace MWScript
|
|
|
|
void execute (Interpreter::Runtime& runtime) override
|
|
|
|
void execute (Interpreter::Runtime& runtime) override
|
|
|
|
{
|
|
|
|
{
|
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
|
|
|
|
|
|
|
if (!actor.getClass().isActor())
|
|
|
|
|
|
|
|
return;
|
|
|
|
MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
|
|
|
MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
|
|
|
creatureStats.getAiSequence().stopCombat();
|
|
|
|
creatureStats.getAiSequence().stopCombat();
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -505,6 +540,9 @@ namespace MWScript
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!actor.getClass().isActor() || actor == MWMechanics::getPlayer())
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
MWMechanics::AiFace facePackage(x, y);
|
|
|
|
MWMechanics::AiFace facePackage(x, y);
|
|
|
|
actor.getClass().getCreatureStats(actor).getAiSequence().stack(facePackage, actor);
|
|
|
|
actor.getClass().getCreatureStats(actor).getAiSequence().stack(facePackage, actor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|