mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-09 21:42:13 +00:00
442 lines
15 KiB
C++
442 lines
15 KiB
C++
|
|
#include "statsextensions.hpp"
|
|
|
|
#include <components/compiler/extensions.hpp>
|
|
|
|
#include <components/interpreter/interpreter.hpp>
|
|
#include <components/interpreter/runtime.hpp>
|
|
#include <components/interpreter/opcodes.hpp>
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
#include "../mwmechanics/npcstats.hpp"
|
|
|
|
#include "interpretercontext.hpp"
|
|
#include "ref.hpp"
|
|
|
|
namespace MWScript
|
|
{
|
|
namespace Stats
|
|
{
|
|
template<class R>
|
|
class OpGetAttribute : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpGetAttribute (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value =
|
|
MWWorld::Class::get (ptr).getCreatureStats (ptr).mAttributes[mIndex].
|
|
getModified();
|
|
|
|
runtime.push (value);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpSetAttribute : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpSetAttribute (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
MWWorld::Class::get (ptr).getCreatureStats (ptr).mAttributes[mIndex].
|
|
setModified (value, 0);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpModAttribute : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpModAttribute (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
value += MWWorld::Class::get (ptr).getCreatureStats (ptr).mAttributes[mIndex].
|
|
getModified();
|
|
|
|
MWWorld::Class::get (ptr).getCreatureStats (ptr).mAttributes[mIndex].
|
|
setModified (value, 0, 100);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpGetDynamic : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpGetDynamic (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
if (mIndex==0 && MWWorld::Class::get (ptr).hasItemHealth (ptr))
|
|
{
|
|
// health is a special case
|
|
Interpreter::Type_Integer value =
|
|
MWWorld::Class::get (ptr).getItemMaxHealth (ptr);
|
|
runtime.push (value);
|
|
|
|
return;
|
|
}
|
|
|
|
Interpreter::Type_Integer value =
|
|
MWWorld::Class::get (ptr).getCreatureStats (ptr).mDynamic[mIndex].
|
|
getCurrent();
|
|
|
|
runtime.push (value);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpSetDynamic : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpSetDynamic (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
MWWorld::Class::get (ptr).getCreatureStats (ptr).mDynamic[mIndex].
|
|
setModified (value, 0);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpModDynamic : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpModDynamic (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer diff = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
MWMechanics::CreatureStats& stats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
Interpreter::Type_Integer current = stats.mDynamic[mIndex].getCurrent();
|
|
|
|
stats.mDynamic[mIndex].setModified (
|
|
diff + stats.mDynamic[mIndex].getModified(), 0);
|
|
|
|
stats.mDynamic[mIndex].setCurrent (diff + current);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpModCurrentDynamic : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpModCurrentDynamic (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer diff = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
MWMechanics::CreatureStats& stats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
Interpreter::Type_Integer current = stats.mDynamic[mIndex].getCurrent();
|
|
|
|
stats.mDynamic[mIndex].setCurrent (diff + current);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpGetDynamicGetRatio : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpGetDynamicGetRatio (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
MWMechanics::CreatureStats& stats = MWWorld::Class::get (ptr).getCreatureStats (ptr);
|
|
|
|
Interpreter::Type_Float value = 0;
|
|
|
|
Interpreter::Type_Float max = stats.mDynamic[mIndex].getModified();
|
|
|
|
if (max>0)
|
|
value = stats.mDynamic[mIndex].getCurrent() / max;
|
|
|
|
runtime.push (value);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpGetSkill : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpGetSkill (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value =
|
|
MWWorld::Class::get (ptr).getNpcStats (ptr).mSkill[mIndex].
|
|
getModified();
|
|
|
|
runtime.push (value);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpSetSkill : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpSetSkill (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
MWWorld::Class::get (ptr).getNpcStats (ptr).mSkill[mIndex].
|
|
setModified (value, 0);
|
|
}
|
|
};
|
|
|
|
template<class R>
|
|
class OpModSkill : public Interpreter::Opcode0
|
|
{
|
|
int mIndex;
|
|
|
|
public:
|
|
|
|
OpModSkill (int index) : mIndex (index) {}
|
|
|
|
virtual void execute (Interpreter::Runtime& runtime)
|
|
{
|
|
MWWorld::Ptr ptr = R()(runtime);
|
|
|
|
Interpreter::Type_Integer value = runtime[0].mInteger;
|
|
runtime.pop();
|
|
|
|
value += MWWorld::Class::get (ptr).getNpcStats (ptr).mSkill[mIndex].
|
|
getModified();
|
|
|
|
MWWorld::Class::get (ptr).getNpcStats (ptr).mSkill[mIndex].
|
|
setModified (value, 0, 100);
|
|
}
|
|
};
|
|
|
|
const int numberOfAttributes = 8;
|
|
|
|
const int opcodeGetAttribute = 0x2000027;
|
|
const int opcodeGetAttributeExplicit = 0x200002f;
|
|
const int opcodeSetAttribute = 0x2000037;
|
|
const int opcodeSetAttributeExplicit = 0x200003f;
|
|
const int opcodeModAttribute = 0x2000047;
|
|
const int opcodeModAttributeExplicit = 0x200004f;
|
|
|
|
const int numberOfDynamics = 3;
|
|
|
|
const int opcodeGetDynamic = 0x2000057;
|
|
const int opcodeGetDynamicExplicit = 0x200005a;
|
|
const int opcodeSetDynamic = 0x200005d;
|
|
const int opcodeSetDynamicExplicit = 0x2000060;
|
|
const int opcodeModDynamic = 0x2000063;
|
|
const int opcodeModDynamicExplicit = 0x2000066;
|
|
const int opcodeModCurrentDynamic = 0x2000069;
|
|
const int opcodeModCurrentDynamicExplicit = 0x200006c;
|
|
const int opcodeGetDynamicGetRatio = 0x200006f;
|
|
const int opcodeGetDynamicGetRatioExplicit = 0x2000072;
|
|
|
|
const int numberOfSkills = 27;
|
|
|
|
const int opcodeGetSkill = 0x200008e;
|
|
const int opcodeGetSkillExplicit = 0x20000a9;
|
|
const int opcodeSetSkill = 0x20000c4;
|
|
const int opcodeSetSkillExplicit = 0x20000df;
|
|
const int opcodeModSkill = 0x20000fa;
|
|
const int opcodeModSkillExplicit = 0x2000115;
|
|
|
|
void registerExtensions (Compiler::Extensions& extensions)
|
|
{
|
|
static const char *attributes[numberOfAttributes] =
|
|
{
|
|
"strength", "intelligence", "willpower", "agility", "speed", "endurance",
|
|
"personality", "luck"
|
|
};
|
|
|
|
static const char *dynamics[numberOfDynamics] =
|
|
{
|
|
"health", "magicka", "fatigue"
|
|
};
|
|
|
|
static const char *skills[numberOfSkills] =
|
|
{
|
|
"block", "armorer", "mediumarmor", "heavyarmor", "bluntweapon",
|
|
"longblade", "axe", "spear", "athletics", "enchant", "destruction",
|
|
"alteration", "illusion", "conjuration", "mysticism",
|
|
"restoration", "alchemy", "unarmored", "security", "sneak",
|
|
"acrobatics", "lightarmor", "shortblade", "marksman",
|
|
"merchantile", "speechcraft", "handtohand"
|
|
};
|
|
|
|
std::string get ("get");
|
|
std::string set ("set");
|
|
std::string mod ("mod");
|
|
std::string modCurrent ("modcurrent");
|
|
std::string getRatio ("getratio");
|
|
|
|
for (int i=0; i<numberOfAttributes; ++i)
|
|
{
|
|
extensions.registerFunction (get + attributes[i], 'l', "",
|
|
opcodeGetAttribute+i, opcodeGetAttributeExplicit+i);
|
|
|
|
extensions.registerInstruction (set + attributes[i], "l",
|
|
opcodeSetAttribute+i, opcodeSetAttributeExplicit+i);
|
|
|
|
extensions.registerInstruction (mod + attributes[i], "l",
|
|
opcodeModAttribute+i, opcodeModAttributeExplicit+i);
|
|
}
|
|
|
|
for (int i=0; i<numberOfDynamics; ++i)
|
|
{
|
|
extensions.registerFunction (get + dynamics[i], 'l', "",
|
|
opcodeGetDynamic+i, opcodeGetDynamicExplicit+i);
|
|
|
|
extensions.registerInstruction (set + dynamics[i], "l",
|
|
opcodeSetDynamic+i, opcodeSetDynamicExplicit+i);
|
|
|
|
extensions.registerInstruction (mod + dynamics[i], "l",
|
|
opcodeModDynamic+i, opcodeModDynamicExplicit+i);
|
|
|
|
extensions.registerInstruction (modCurrent + dynamics[i], "l",
|
|
opcodeModCurrentDynamic+i, opcodeModCurrentDynamicExplicit+i);
|
|
|
|
extensions.registerFunction (get + dynamics[i] + getRatio, 'f', "",
|
|
opcodeGetDynamicGetRatio+i, opcodeGetDynamicGetRatioExplicit+i);
|
|
}
|
|
|
|
for (int i=0; i<numberOfSkills; ++i)
|
|
{
|
|
extensions.registerFunction (get + skills[i], 'l', "",
|
|
opcodeGetSkill+i, opcodeGetSkillExplicit+i);
|
|
|
|
extensions.registerInstruction (set + skills[i], "l",
|
|
opcodeSetSkill+i, opcodeSetSkillExplicit+i);
|
|
|
|
extensions.registerInstruction (mod + skills[i], "l",
|
|
opcodeModSkill+i, opcodeModSkillExplicit+i);
|
|
}
|
|
}
|
|
|
|
void installOpcodes (Interpreter::Interpreter& interpreter)
|
|
{
|
|
for (int i=0; i<numberOfAttributes; ++i)
|
|
{
|
|
interpreter.installSegment5 (opcodeGetAttribute+i, new OpGetAttribute<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeGetAttributeExplicit+i,
|
|
new OpGetAttribute<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeSetAttribute+i, new OpSetAttribute<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeSetAttributeExplicit+i,
|
|
new OpSetAttribute<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeModAttribute+i, new OpModAttribute<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeModAttributeExplicit+i,
|
|
new OpModAttribute<ExplicitRef> (i));
|
|
}
|
|
|
|
for (int i=0; i<numberOfDynamics; ++i)
|
|
{
|
|
interpreter.installSegment5 (opcodeGetDynamic+i, new OpGetDynamic<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeGetDynamicExplicit+i,
|
|
new OpGetDynamic<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeSetDynamic+i, new OpSetDynamic<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeSetDynamicExplicit+i,
|
|
new OpSetDynamic<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeModDynamic+i, new OpModDynamic<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeModDynamicExplicit+i,
|
|
new OpModDynamic<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeModCurrentDynamic+i,
|
|
new OpModCurrentDynamic<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeModCurrentDynamicExplicit+i,
|
|
new OpModCurrentDynamic<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeGetDynamicGetRatio+i,
|
|
new OpGetDynamicGetRatio<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeGetDynamicGetRatioExplicit+i,
|
|
new OpGetDynamicGetRatio<ExplicitRef> (i));
|
|
}
|
|
|
|
for (int i=0; i<numberOfSkills; ++i)
|
|
{
|
|
interpreter.installSegment5 (opcodeGetSkill+i, new OpGetSkill<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeGetSkillExplicit+i, new OpGetSkill<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeSetSkill+i, new OpSetSkill<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeSetSkillExplicit+i, new OpSetSkill<ExplicitRef> (i));
|
|
|
|
interpreter.installSegment5 (opcodeModSkill+i, new OpModSkill<ImplicitRef> (i));
|
|
interpreter.installSegment5 (opcodeModSkillExplicit+i, new OpModSkill<ExplicitRef> (i));
|
|
}
|
|
}
|
|
}
|
|
}
|