2010-08-22 12:49:25 +00:00
|
|
|
#include "aiextensions.hpp"
|
|
|
|
|
2015-12-06 22:40:03 +00:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <iostream>
|
|
|
|
|
2018-07-12 20:08:17 +00:00
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Include additional headers for multiplayer purposes
|
|
|
|
*/
|
|
|
|
#include <components/openmw-mp/Log.hpp>
|
|
|
|
#include "../mwmp/Main.hpp"
|
|
|
|
#include "../mwmp/Networking.hpp"
|
|
|
|
#include "../mwmp/ActorList.hpp"
|
|
|
|
#include "../mwmp/MechanicsHelper.hpp"
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
#include <components/compiler/extensions.hpp>
|
2013-08-07 00:38:41 +00:00
|
|
|
#include <components/compiler/opcodes.hpp>
|
2010-08-22 12:49:25 +00:00
|
|
|
|
|
|
|
#include <components/interpreter/interpreter.hpp>
|
|
|
|
#include <components/interpreter/runtime.hpp>
|
|
|
|
#include <components/interpreter/opcodes.hpp>
|
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
2015-12-06 22:40:03 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2012-07-03 10:30:50 +00:00
|
|
|
|
2012-06-16 14:31:32 +00:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
2012-11-30 00:16:16 +00:00
|
|
|
#include "../mwmechanics/aiactivate.hpp"
|
|
|
|
#include "../mwmechanics/aiescort.hpp"
|
|
|
|
#include "../mwmechanics/aifollow.hpp"
|
|
|
|
#include "../mwmechanics/aitravel.hpp"
|
|
|
|
#include "../mwmechanics/aiwander.hpp"
|
2016-07-09 00:16:47 +00:00
|
|
|
#include "../mwmechanics/aiface.hpp"
|
2012-06-16 14:31:32 +00:00
|
|
|
|
2013-09-10 14:16:13 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
2015-12-06 22:40:03 +00:00
|
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
2013-09-10 14:16:13 +00:00
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
#include "interpretercontext.hpp"
|
2010-12-31 18:09:25 +00:00
|
|
|
#include "ref.hpp"
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2013-11-18 22:03:44 +00:00
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
namespace MWScript
|
|
|
|
{
|
|
|
|
namespace Ai
|
|
|
|
{
|
2013-01-03 01:34:31 +00:00
|
|
|
template<class R>
|
|
|
|
class OpAiActivate : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2013-01-03 01:34:31 +00:00
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
|
|
|
std::string objectID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2013-01-03 01:34:31 +00:00
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2013-01-03 01:34:31 +00:00
|
|
|
MWMechanics::AiActivate activatePackage(objectID);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(activatePackage, ptr);
|
2013-01-03 01:34:31 +00:00
|
|
|
std::cout << "AiActivate" << std::endl;
|
|
|
|
}
|
|
|
|
};
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2010-12-31 18:09:25 +00:00
|
|
|
template<class R>
|
2010-08-22 12:49:25 +00:00
|
|
|
class OpAiTravel : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
2010-12-31 18:09:25 +00:00
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2012-11-30 00:16:16 +00:00
|
|
|
MWMechanics::AiTravel travelPackage(x, y, z);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(travelPackage, ptr);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
std::cout << "AiTravel: " << x << ", " << y << ", " << z << std::endl;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-12-31 18:09:25 +00:00
|
|
|
template<class R>
|
2010-08-22 12:49:25 +00:00
|
|
|
class OpAiEscort : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
2010-12-31 18:09:25 +00:00
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2012-11-30 00:16:16 +00:00
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float duration = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
MWMechanics::AiEscort escortPackage(actorID, static_cast<int>(duration), x, y, z);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
|
|
|
std::cout << "AiEscort: " << x << ", " << y << ", " << z << ", " << duration
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpAiEscortCell : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
std::string cellID = runtime.getStringLiteral (runtime[0].mInteger);
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float duration = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2011-11-21 13:16:20 +00:00
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
2010-08-22 12:49:25 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2015-12-06 22:40:03 +00:00
|
|
|
if (cellID.empty())
|
|
|
|
throw std::runtime_error("AiEscortCell: no cell ID given");
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Cell>().find(cellID);
|
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
MWMechanics::AiEscort escortPackage(actorID, cellID, static_cast<int>(duration), x, y, z);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(escortPackage, ptr);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
std::cout << "AiEscort: " << x << ", " << y << ", " << z << ", " << duration
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-12-31 18:09:25 +00:00
|
|
|
template<class R>
|
2010-08-22 12:49:25 +00:00
|
|
|
class OpGetAiPackageDone : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
2010-12-31 18:09:25 +00:00
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2014-05-22 18:37:22 +00:00
|
|
|
Interpreter::Type_Integer value = ptr.getClass().getCreatureStats (ptr).getAiSequence().isPackageDone();
|
2010-08-22 12:49:25 +00:00
|
|
|
|
|
|
|
runtime.push (value);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-06-16 14:06:15 +00:00
|
|
|
template<class R>
|
|
|
|
class OpAiWander : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
Interpreter::Type_Integer range = static_cast<Interpreter::Type_Integer>(runtime[0].mFloat);
|
2012-06-16 14:06:15 +00:00
|
|
|
runtime.pop();
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
Interpreter::Type_Integer duration = static_cast<Interpreter::Type_Integer>(runtime[0].mFloat);
|
2012-06-16 14:06:15 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
Interpreter::Type_Integer time = static_cast<Interpreter::Type_Integer>(runtime[0].mFloat);
|
2012-06-16 14:06:15 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2018-06-16 18:11:10 +00:00
|
|
|
// Chance for Idle is unused
|
|
|
|
if (arg0)
|
|
|
|
{
|
|
|
|
--arg0;
|
|
|
|
runtime.pop();
|
|
|
|
}
|
|
|
|
|
2014-06-12 21:27:04 +00:00
|
|
|
std::vector<unsigned char> idleList;
|
2013-05-24 11:49:20 +00:00
|
|
|
bool repeat = false;
|
2013-01-15 11:40:44 +00:00
|
|
|
|
2018-06-16 18:11:10 +00:00
|
|
|
// Chances for Idle2-Idle9
|
|
|
|
for(int i=2; i<=9 && arg0; ++i)
|
2013-01-15 11:40:44 +00:00
|
|
|
{
|
2013-05-24 11:49:20 +00:00
|
|
|
if(!repeat)
|
|
|
|
repeat = true;
|
2013-05-13 09:27:36 +00:00
|
|
|
Interpreter::Type_Integer idleValue = runtime[0].mInteger;
|
2014-06-12 21:27:04 +00:00
|
|
|
idleValue = std::min(255, std::max(0, idleValue));
|
2013-01-03 01:34:31 +00:00
|
|
|
idleList.push_back(idleValue);
|
|
|
|
runtime.pop();
|
2013-01-15 11:40:44 +00:00
|
|
|
--arg0;
|
2012-11-30 00:16:16 +00:00
|
|
|
}
|
|
|
|
|
2013-05-24 11:49:20 +00:00
|
|
|
if(arg0)
|
|
|
|
{
|
2013-05-24 13:03:46 +00:00
|
|
|
repeat = runtime[0].mInteger != 0;
|
2013-05-24 11:49:20 +00:00
|
|
|
runtime.pop();
|
|
|
|
--arg0;
|
|
|
|
}
|
|
|
|
|
2012-06-16 14:06:15 +00:00
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2013-05-24 11:49:20 +00:00
|
|
|
MWMechanics::AiWander wanderPackage(range, duration, time, idleList, repeat);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(wanderPackage, ptr);
|
2012-06-16 14:06:15 +00:00
|
|
|
}
|
|
|
|
};
|
2010-08-22 12:49:25 +00:00
|
|
|
|
2012-06-16 14:31:32 +00:00
|
|
|
template<class R>
|
2012-11-23 20:14:32 +00:00
|
|
|
class OpGetAiSetting : public Interpreter::Opcode0
|
2012-06-16 14:31:32 +00:00
|
|
|
{
|
2012-11-23 20:14:32 +00:00
|
|
|
int mIndex;
|
2012-06-16 14:31:32 +00:00
|
|
|
public:
|
2012-11-23 20:14:32 +00:00
|
|
|
OpGetAiSetting(int index) : mIndex(index) {}
|
2012-06-16 14:31:32 +00:00
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
2014-05-22 18:37:22 +00:00
|
|
|
runtime.push(ptr.getClass().getCreatureStats (ptr).getAiSetting (
|
2014-01-05 00:34:35 +00:00
|
|
|
(MWMechanics::CreatureStats::AiSetting)mIndex).getModified());
|
2012-06-16 14:31:32 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
template<class R>
|
2012-11-23 20:14:32 +00:00
|
|
|
class OpModAiSetting : public Interpreter::Opcode0
|
2012-06-16 14:31:32 +00:00
|
|
|
{
|
2012-11-23 20:14:32 +00:00
|
|
|
int mIndex;
|
2012-06-16 14:31:32 +00:00
|
|
|
public:
|
2012-11-23 20:14:32 +00:00
|
|
|
OpModAiSetting(int index) : mIndex(index) {}
|
2012-06-16 14:31:32 +00:00
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
|
|
runtime.pop();
|
|
|
|
|
2014-01-05 00:34:35 +00:00
|
|
|
MWMechanics::CreatureStats::AiSetting setting
|
|
|
|
= MWMechanics::CreatureStats::AiSetting(mIndex);
|
|
|
|
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).setAiSetting (setting,
|
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSetting (setting).getBase() + value);
|
2012-06-16 14:31:32 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
template<class R>
|
2012-11-23 20:14:32 +00:00
|
|
|
class OpSetAiSetting : public Interpreter::Opcode0
|
2012-06-16 14:31:32 +00:00
|
|
|
{
|
2012-11-23 20:14:32 +00:00
|
|
|
int mIndex;
|
2012-06-16 14:31:32 +00:00
|
|
|
public:
|
2012-11-23 20:14:32 +00:00
|
|
|
OpSetAiSetting(int index) : mIndex(index) {}
|
2012-06-16 14:31:32 +00:00
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
|
|
runtime.pop();
|
|
|
|
|
2014-01-05 00:34:35 +00:00
|
|
|
MWMechanics::CreatureStats::AiSetting setting = (MWMechanics::CreatureStats::AiSetting)mIndex;
|
|
|
|
|
|
|
|
MWMechanics::Stat<int> stat = ptr.getClass().getCreatureStats(ptr).getAiSetting(setting);
|
|
|
|
stat.setModified(value, 0);
|
|
|
|
ptr.getClass().getCreatureStats(ptr).setAiSetting(setting, stat);
|
2012-06-16 14:31:32 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-11-30 00:16:16 +00:00
|
|
|
template<class R>
|
|
|
|
class OpAiFollow : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float duration = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2014-05-16 10:11:34 +00:00
|
|
|
MWMechanics::AiFollow followPackage(actorID, duration, x, y ,z);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2013-01-03 01:34:31 +00:00
|
|
|
std::cout << "AiFollow: " << actorID << ", " << x << ", " << y << ", " << z << ", " << duration
|
2012-11-30 00:16:16 +00:00
|
|
|
<< std::endl;
|
2018-07-12 20:08:17 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
2018-07-13 21:07:15 +00:00
|
|
|
Send ActorAI packets when an actor becomes a follower, regardless of whether we're
|
|
|
|
the cell authority or not; the server can decide if it wants to comply with them by
|
|
|
|
forwarding them to the cell authority
|
2018-07-12 20:08:17 +00:00
|
|
|
*/
|
|
|
|
MWWorld::Ptr targetPtr = MWBase::Environment::get().getWorld()->searchPtr(actorID, true);
|
|
|
|
|
|
|
|
if (targetPtr)
|
|
|
|
{
|
|
|
|
mwmp::ActorList *actorList = mwmp::Main::get().getNetworking()->getActorList();
|
|
|
|
actorList->reset();
|
|
|
|
actorList->cell = *ptr.getCell()->getCell();
|
2018-07-13 22:00:27 +00:00
|
|
|
actorList->addAiActor(ptr, targetPtr, mwmp::BaseActorList::FOLLOW);
|
2018-07-12 20:08:17 +00:00
|
|
|
actorList->sendAiActors();
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
2012-11-30 00:16:16 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpAiFollowCell : public Interpreter::Opcode1
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime, unsigned int arg0)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
std::string cellID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float duration = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float z = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
// discard additional arguments (reset), because we have no idea what they mean.
|
|
|
|
for (unsigned int i=0; i<arg0; ++i) runtime.pop();
|
|
|
|
|
2014-05-16 10:11:34 +00:00
|
|
|
MWMechanics::AiFollow followPackage(actorID, cellID, duration, x, y ,z);
|
2014-05-22 18:37:22 +00:00
|
|
|
ptr.getClass().getCreatureStats (ptr).getAiSequence().stack(followPackage, ptr);
|
2012-11-30 00:16:16 +00:00
|
|
|
std::cout << "AiFollow: " << actorID << ", " << x << ", " << y << ", " << z << ", " << duration
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpGetCurrentAIPackage : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
|
2014-05-22 18:37:22 +00:00
|
|
|
Interpreter::Type_Integer value = ptr.getClass().getCreatureStats (ptr).getAiSequence().getLastRunTypeId();
|
2012-11-30 00:16:16 +00:00
|
|
|
|
|
|
|
runtime.push (value);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
2014-01-06 23:51:09 +00:00
|
|
|
class OpGetDetected : public Interpreter::Opcode0
|
2012-11-30 00:16:16 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2014-01-06 23:51:09 +00:00
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
2012-11-30 00:16:16 +00:00
|
|
|
{
|
2017-09-18 17:46:57 +00:00
|
|
|
MWWorld::Ptr observer = R()(runtime, false); // required=false
|
|
|
|
|
2012-11-30 00:16:16 +00:00
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2014-01-06 23:51:09 +00:00
|
|
|
MWWorld::Ptr actor = MWBase::Environment::get().getWorld()->getPtr(actorID, true);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
2017-09-18 17:46:57 +00:00
|
|
|
Interpreter::Type_Integer value = MWBase::Environment::get().getMechanicsManager()->isActorDetected(actor, observer);
|
2012-11-30 00:16:16 +00:00
|
|
|
|
|
|
|
runtime.push (value);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-09-10 14:16:13 +00:00
|
|
|
template<class R>
|
|
|
|
class OpGetLineOfSight : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
|
|
|
|
MWWorld::Ptr source = R()(runtime);
|
|
|
|
|
|
|
|
std::string actorID = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2014-06-19 02:53:25 +00:00
|
|
|
|
2013-09-10 14:16:13 +00:00
|
|
|
MWWorld::Ptr dest = MWBase::Environment::get().getWorld()->getPtr(actorID,true);
|
|
|
|
bool value = false;
|
2014-06-19 02:53:25 +00:00
|
|
|
if(dest != MWWorld::Ptr() && source.getClass().isActor() && dest.getClass().isActor())
|
2013-09-10 14:16:13 +00:00
|
|
|
{
|
|
|
|
value = MWBase::Environment::get().getWorld()->getLOS(source,dest);
|
|
|
|
}
|
|
|
|
runtime.push (value);
|
|
|
|
}
|
|
|
|
};
|
2012-06-16 14:31:32 +00:00
|
|
|
|
2014-01-07 00:12:37 +00:00
|
|
|
template<class R>
|
|
|
|
class OpGetTarget : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void execute (Interpreter::Runtime &runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
|
|
|
std::string testedTargetId = runtime.getStringLiteral (runtime[0].mInteger);
|
|
|
|
runtime.pop();
|
|
|
|
|
2014-05-22 18:37:22 +00:00
|
|
|
const MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
2014-01-07 00:12:37 +00:00
|
|
|
|
|
|
|
bool targetsAreEqual = false;
|
2014-05-18 16:13:46 +00:00
|
|
|
MWWorld::Ptr targetPtr;
|
|
|
|
if (creatureStats.getAiSequence().getCombatTarget (targetPtr))
|
2014-01-07 00:12:37 +00:00
|
|
|
{
|
2015-11-30 14:26:43 +00:00
|
|
|
if (!targetPtr.isEmpty() && targetPtr.getCellRef().getRefId() == testedTargetId)
|
2014-01-07 00:12:37 +00:00
|
|
|
targetsAreEqual = true;
|
|
|
|
}
|
|
|
|
runtime.push(int(targetsAreEqual));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpStartCombat : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void execute (Interpreter::Runtime &runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
2014-01-18 10:55:17 +00:00
|
|
|
std::string targetID = runtime.getStringLiteral (runtime[0].mInteger);
|
2014-01-07 00:12:37 +00:00
|
|
|
runtime.pop();
|
|
|
|
|
2014-05-05 22:13:31 +00:00
|
|
|
MWWorld::Ptr target = MWBase::Environment::get().getWorld()->getPtr(targetID, true);
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->startCombat(actor, target);
|
2018-07-13 21:07:15 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Send ActorAI packets when an actor starts combat, regardless of whether we're the
|
|
|
|
cell authority or not; the server can decide if it wants to comply with them by
|
|
|
|
forwarding them to the cell authority
|
|
|
|
*/
|
|
|
|
if (target)
|
|
|
|
{
|
|
|
|
mwmp::ActorList *actorList = mwmp::Main::get().getNetworking()->getActorList();
|
|
|
|
actorList->reset();
|
|
|
|
actorList->cell = *actor.getCell()->getCell();
|
2018-07-13 22:00:27 +00:00
|
|
|
actorList->addAiActor(actor, target, mwmp::BaseActorList::COMBAT);
|
2018-07-13 21:07:15 +00:00
|
|
|
actorList->sendAiActors();
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
2014-01-07 00:12:37 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class R>
|
|
|
|
class OpStopCombat : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
2014-05-22 18:37:22 +00:00
|
|
|
MWMechanics::CreatureStats& creatureStats = actor.getClass().getCreatureStats(actor);
|
2014-01-07 00:12:37 +00:00
|
|
|
creatureStats.getAiSequence().stopCombat();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-11-18 22:03:44 +00:00
|
|
|
class OpToggleAI : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
|
|
{
|
2014-03-26 18:55:52 +00:00
|
|
|
bool enabled = MWBase::Environment::get().getMechanicsManager()->toggleAI();
|
|
|
|
|
2015-03-03 22:46:53 +00:00
|
|
|
runtime.getContext().report (enabled ? "AI -> On" : "AI -> Off");
|
2013-11-18 22:03:44 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-24 13:09:13 +00:00
|
|
|
template <class R>
|
|
|
|
class OpFace : public Interpreter::Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void execute(Interpreter::Runtime& runtime)
|
|
|
|
{
|
2016-07-09 00:16:47 +00:00
|
|
|
MWWorld::Ptr actor = R()(runtime);
|
|
|
|
|
|
|
|
Interpreter::Type_Float x = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
Interpreter::Type_Float y = runtime[0].mFloat;
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
MWMechanics::AiFace facePackage(x, y);
|
|
|
|
actor.getClass().getCreatureStats(actor).getAiSequence().stack(facePackage, actor);
|
2014-06-24 13:09:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-08-22 12:49:25 +00:00
|
|
|
void installOpcodes (Interpreter::Interpreter& interpreter)
|
|
|
|
{
|
2013-08-07 00:38:41 +00:00
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAIActivate, new OpAiActivate<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAIActivateExplicit, new OpAiActivate<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiTravel, new OpAiTravel<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiTravelExplicit, new OpAiTravel<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiEscort, new OpAiEscort<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiEscortExplicit, new OpAiEscort<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiEscortCell, new OpAiEscortCell<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiEscortCellExplicit, new OpAiEscortCell<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiWander, new OpAiWander<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiWanderExplicit, new OpAiWander<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiFollow, new OpAiFollow<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiFollowExplicit, new OpAiFollow<ExplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiFollowCell, new OpAiFollowCell<ImplicitRef>);
|
|
|
|
interpreter.installSegment3 (Compiler::Ai::opcodeAiFollowCellExplicit, new OpAiFollowCell<ExplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetAiPackageDone, new OpGetAiPackageDone<ImplicitRef>);
|
|
|
|
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetAiPackageDoneExplicit,
|
2010-12-31 18:09:25 +00:00
|
|
|
new OpGetAiPackageDone<ExplicitRef>);
|
2013-08-07 00:38:41 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetCurrentAiPackage, new OpGetCurrentAIPackage<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetCurrentAiPackageExplicit, new OpGetCurrentAIPackage<ExplicitRef>);
|
2014-01-06 23:51:09 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetDetected, new OpGetDetected<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetDetectedExplicit, new OpGetDetected<ExplicitRef>);
|
2013-09-10 14:16:13 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetLineOfSight, new OpGetLineOfSight<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetLineOfSightExplicit, new OpGetLineOfSight<ExplicitRef>);
|
2014-01-07 00:12:37 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetTarget, new OpGetTarget<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetTargetExplicit, new OpGetTarget<ExplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeStartCombat, new OpStartCombat<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeStartCombatExplicit, new OpStartCombat<ExplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeStopCombat, new OpStopCombat<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeStopCombatExplicit, new OpStopCombat<ExplicitRef>);
|
2016-03-12 19:32:42 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeToggleAI, new OpToggleAI);
|
2013-09-10 14:16:13 +00:00
|
|
|
|
2013-08-07 00:38:41 +00:00
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetHello, new OpSetAiSetting<ImplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetHelloExplicit, new OpSetAiSetting<ExplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetFight, new OpSetAiSetting<ImplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetFightExplicit, new OpSetAiSetting<ExplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetFlee, new OpSetAiSetting<ImplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetFleeExplicit, new OpSetAiSetting<ExplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetAlarm, new OpSetAiSetting<ImplicitRef>(3));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeSetAlarmExplicit, new OpSetAiSetting<ExplicitRef>(3));
|
|
|
|
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModHello, new OpModAiSetting<ImplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModHelloExplicit, new OpModAiSetting<ExplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModFight, new OpModAiSetting<ImplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModFightExplicit, new OpModAiSetting<ExplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModFlee, new OpModAiSetting<ImplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModFleeExplicit, new OpModAiSetting<ExplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModAlarm, new OpModAiSetting<ImplicitRef>(3));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeModAlarmExplicit, new OpModAiSetting<ExplicitRef>(3));
|
|
|
|
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetHello, new OpGetAiSetting<ImplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetHelloExplicit, new OpGetAiSetting<ExplicitRef>(0));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetFight, new OpGetAiSetting<ImplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetFightExplicit, new OpGetAiSetting<ExplicitRef>(1));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetFlee, new OpGetAiSetting<ImplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetFleeExplicit, new OpGetAiSetting<ExplicitRef>(2));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetAlarm, new OpGetAiSetting<ImplicitRef>(3));
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeGetAlarmExplicit, new OpGetAiSetting<ExplicitRef>(3));
|
2014-06-24 13:09:13 +00:00
|
|
|
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeFace, new OpFace<ImplicitRef>);
|
|
|
|
interpreter.installSegment5 (Compiler::Ai::opcodeFaceExplicit, new OpFace<ExplicitRef>);
|
2010-08-22 12:49:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|