2022-03-25 20:03:13 +00:00
|
|
|
#include "stats.hpp"
|
|
|
|
|
2022-07-11 16:40:06 +02:00
|
|
|
#include <algorithm>
|
2022-03-25 20:03:13 +00:00
|
|
|
#include <memory>
|
|
|
|
#include <optional>
|
|
|
|
#include <string_view>
|
|
|
|
#include <variant>
|
|
|
|
|
2022-09-05 19:35:15 +02:00
|
|
|
#include <components/esm3/loadclas.hpp>
|
2022-03-25 20:03:13 +00:00
|
|
|
#include <components/lua/luastate.hpp>
|
|
|
|
|
2022-08-17 19:09:38 +02:00
|
|
|
#include "context.hpp"
|
2022-03-25 20:03:13 +00:00
|
|
|
#include "localscripts.hpp"
|
|
|
|
#include "luamanagerimp.hpp"
|
|
|
|
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwmechanics/npcstats.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
template <class T>
|
|
|
|
auto addIndexedAccessor(int index)
|
|
|
|
{
|
|
|
|
return sol::overload([index](MWLua::LocalScripts::SelfObject& o) { return T::create(&o, index); },
|
|
|
|
[index](const MWLua::LObject& o) { return T::create(o, index); },
|
|
|
|
[index](const MWLua::GObject& o) { return T::create(o, index); });
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T, class G>
|
2022-07-11 16:40:06 +02:00
|
|
|
void addProp(const MWLua::Context& context, sol::usertype<T>& type, std::string_view prop, G getter)
|
2022-03-25 20:03:13 +00:00
|
|
|
{
|
2022-07-11 16:40:06 +02:00
|
|
|
type[prop] = sol::property([=](const T& stat) { return stat.get(context, prop, getter); },
|
|
|
|
[=](const T& stat, const sol::object& value) { stat.cache(context, prop, value); });
|
2022-03-25 20:03:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
using SelfObject = MWLua::LocalScripts::SelfObject;
|
|
|
|
using StatObject = std::variant<SelfObject*, MWLua::LObject, MWLua::GObject>;
|
2022-10-10 22:40:33 +02:00
|
|
|
SelfObject* asSelfObject(const StatObject& obj)
|
|
|
|
{
|
|
|
|
if (!std::holds_alternative<SelfObject*>(obj))
|
|
|
|
throw std::runtime_error("Changing stats allowed only in local scripts for 'openmw.self'.");
|
|
|
|
return std::get<SelfObject*>(obj);
|
|
|
|
}
|
2022-03-25 20:03:13 +00:00
|
|
|
|
|
|
|
const MWLua::Object* getObject(const StatObject& obj)
|
|
|
|
{
|
|
|
|
return std::visit(
|
|
|
|
[](auto&& variant) -> const MWLua::Object* {
|
|
|
|
using T = std::decay_t<decltype(variant)>;
|
|
|
|
if constexpr (std::is_same_v<T, SelfObject*>)
|
|
|
|
return variant;
|
|
|
|
else if constexpr (std::is_same_v<T, MWLua::LObject>)
|
|
|
|
return &variant;
|
|
|
|
else if constexpr (std::is_same_v<T, MWLua::GObject>)
|
|
|
|
return &variant;
|
|
|
|
},
|
|
|
|
obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class G>
|
|
|
|
sol::object getValue(const MWLua::Context& context, const StatObject& obj, SelfObject::CachedStat::Setter setter,
|
|
|
|
int index, std::string_view prop, G getter)
|
|
|
|
{
|
|
|
|
return std::visit(
|
|
|
|
[&](auto&& variant) {
|
|
|
|
using T = std::decay_t<decltype(variant)>;
|
|
|
|
if constexpr (std::is_same_v<T, SelfObject*>)
|
|
|
|
{
|
|
|
|
auto it = variant->mStatsCache.find({ setter, index, prop });
|
|
|
|
if (it != variant->mStatsCache.end())
|
|
|
|
return it->second;
|
|
|
|
return sol::make_object(context.mLua->sol(), getter(variant));
|
|
|
|
}
|
|
|
|
else if constexpr (std::is_same_v<T, MWLua::LObject>)
|
|
|
|
return sol::make_object(context.mLua->sol(), getter(&variant));
|
|
|
|
else if constexpr (std::is_same_v<T, MWLua::GObject>)
|
|
|
|
return sol::make_object(context.mLua->sol(), getter(&variant));
|
|
|
|
},
|
|
|
|
obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace MWLua
|
|
|
|
{
|
2022-04-11 01:04:55 +02:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
class StatUpdateAction final : public LuaManager::Action
|
|
|
|
{
|
|
|
|
ObjectId mId;
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2022-04-11 01:04:55 +02:00
|
|
|
public:
|
|
|
|
StatUpdateAction(LuaUtil::LuaState* state, ObjectId id)
|
|
|
|
: Action(state)
|
|
|
|
, mId(id)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-12-24 02:25:39 +01:00
|
|
|
void apply() const override
|
2022-04-11 01:04:55 +02:00
|
|
|
{
|
2022-12-23 19:57:12 +01:00
|
|
|
LObject obj(mId);
|
2022-04-11 01:04:55 +02:00
|
|
|
LocalScripts* scripts = obj.ptr().getRefData().getLuaScripts();
|
|
|
|
if (scripts)
|
|
|
|
scripts->applyStatsCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string toString() const override { return "StatUpdateAction"; }
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
class LevelStat
|
|
|
|
{
|
|
|
|
StatObject mObject;
|
|
|
|
|
|
|
|
LevelStat(StatObject object)
|
|
|
|
: mObject(std::move(object))
|
|
|
|
{
|
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
public:
|
|
|
|
sol::object getCurrent(const Context& context) const
|
|
|
|
{
|
|
|
|
return getValue(context, mObject, &LevelStat::setValue, 0, "current", [](const MWLua::Object* obj) {
|
|
|
|
const auto& ptr = obj->ptr();
|
|
|
|
return ptr.getClass().getCreatureStats(ptr).getLevel();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void setCurrent(const Context& context, const sol::object& value) const
|
|
|
|
{
|
2022-10-10 22:40:33 +02:00
|
|
|
SelfObject* obj = asSelfObject(mObject);
|
2022-03-25 20:03:13 +00:00
|
|
|
if (obj->mStatsCache.empty())
|
|
|
|
context.mLuaManager->addAction(std::make_unique<StatUpdateAction>(context.mLua, obj->id()));
|
|
|
|
obj->mStatsCache[SelfObject::CachedStat{ &LevelStat::setValue, 0, "current" }] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
sol::object getProgress(const Context& context) const
|
|
|
|
{
|
|
|
|
const auto& ptr = getObject(mObject)->ptr();
|
|
|
|
if (!ptr.getClass().isNpc())
|
|
|
|
return sol::nil;
|
|
|
|
return sol::make_object(context.mLua->sol(), ptr.getClass().getNpcStats(ptr).getLevelProgress());
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::optional<LevelStat> create(StatObject object, int index)
|
|
|
|
{
|
|
|
|
if (!getObject(object)->ptr().getClass().isActor())
|
|
|
|
return {};
|
|
|
|
return LevelStat{ std::move(object) };
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setValue(int, std::string_view prop, const MWWorld::Ptr& ptr, const sol::object& value)
|
|
|
|
{
|
|
|
|
auto& stats = ptr.getClass().getCreatureStats(ptr);
|
|
|
|
if (prop == "current")
|
|
|
|
stats.setLevel(value.as<int>());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class DynamicStat
|
|
|
|
{
|
|
|
|
StatObject mObject;
|
|
|
|
int mIndex;
|
|
|
|
|
|
|
|
DynamicStat(StatObject object, int index)
|
|
|
|
: mObject(std::move(object))
|
|
|
|
, mIndex(index)
|
|
|
|
{
|
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
public:
|
|
|
|
template <class G>
|
|
|
|
sol::object get(const Context& context, std::string_view prop, G getter) const
|
|
|
|
{
|
2022-06-12 08:00:11 +00:00
|
|
|
return getValue(
|
|
|
|
context, mObject, &DynamicStat::setValue, mIndex, prop, [this, getter](const MWLua::Object* obj) {
|
2022-03-25 20:03:13 +00:00
|
|
|
const auto& ptr = obj->ptr();
|
|
|
|
return (ptr.getClass().getCreatureStats(ptr).getDynamic(mIndex).*getter)();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::optional<DynamicStat> create(StatObject object, int index)
|
|
|
|
{
|
|
|
|
if (!getObject(object)->ptr().getClass().isActor())
|
|
|
|
return {};
|
|
|
|
return DynamicStat{ std::move(object), index };
|
|
|
|
}
|
|
|
|
|
|
|
|
void cache(const Context& context, std::string_view prop, const sol::object& value) const
|
|
|
|
{
|
2022-10-10 22:40:33 +02:00
|
|
|
SelfObject* obj = asSelfObject(mObject);
|
2022-03-25 20:03:13 +00:00
|
|
|
if (obj->mStatsCache.empty())
|
|
|
|
context.mLuaManager->addAction(std::make_unique<StatUpdateAction>(context.mLua, obj->id()));
|
|
|
|
obj->mStatsCache[SelfObject::CachedStat{ &DynamicStat::setValue, mIndex, prop }] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setValue(int index, std::string_view prop, const MWWorld::Ptr& ptr, const sol::object& value)
|
|
|
|
{
|
|
|
|
auto& stats = ptr.getClass().getCreatureStats(ptr);
|
|
|
|
auto stat = stats.getDynamic(index);
|
|
|
|
float floatValue = value.as<float>();
|
|
|
|
if (prop == "base")
|
|
|
|
stat.setBase(floatValue);
|
|
|
|
else if (prop == "current")
|
|
|
|
stat.setCurrent(floatValue, true, true);
|
|
|
|
else if (prop == "modifier")
|
|
|
|
stat.setModifier(floatValue);
|
|
|
|
stats.setDynamic(index, stat);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class AttributeStat
|
|
|
|
{
|
|
|
|
StatObject mObject;
|
|
|
|
int mIndex;
|
|
|
|
|
|
|
|
AttributeStat(StatObject object, int index)
|
|
|
|
: mObject(std::move(object))
|
|
|
|
, mIndex(index)
|
|
|
|
{
|
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
public:
|
|
|
|
template <class G>
|
|
|
|
sol::object get(const Context& context, std::string_view prop, G getter) const
|
|
|
|
{
|
2022-06-12 08:00:11 +00:00
|
|
|
return getValue(
|
|
|
|
context, mObject, &AttributeStat::setValue, mIndex, prop, [this, getter](const MWLua::Object* obj) {
|
2022-03-25 20:03:13 +00:00
|
|
|
const auto& ptr = obj->ptr();
|
|
|
|
return (ptr.getClass().getCreatureStats(ptr).getAttribute(mIndex).*getter)();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-07-11 16:40:06 +02:00
|
|
|
float getModified(const Context& context) const
|
|
|
|
{
|
|
|
|
auto base = get(context, "base", &MWMechanics::AttributeValue::getBase).as<float>();
|
|
|
|
auto damage = get(context, "damage", &MWMechanics::AttributeValue::getDamage).as<float>();
|
|
|
|
auto modifier = get(context, "modifier", &MWMechanics::AttributeValue::getModifier).as<float>();
|
|
|
|
return std::max(0.f, base - damage + modifier); // Should match AttributeValue::getModified
|
|
|
|
}
|
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
static std::optional<AttributeStat> create(StatObject object, int index)
|
|
|
|
{
|
|
|
|
if (!getObject(object)->ptr().getClass().isActor())
|
|
|
|
return {};
|
|
|
|
return AttributeStat{ std::move(object), index };
|
|
|
|
}
|
|
|
|
|
|
|
|
void cache(const Context& context, std::string_view prop, const sol::object& value) const
|
|
|
|
{
|
2022-10-10 22:40:33 +02:00
|
|
|
SelfObject* obj = asSelfObject(mObject);
|
2022-03-25 20:03:13 +00:00
|
|
|
if (obj->mStatsCache.empty())
|
|
|
|
context.mLuaManager->addAction(std::make_unique<StatUpdateAction>(context.mLua, obj->id()));
|
|
|
|
obj->mStatsCache[SelfObject::CachedStat{ &AttributeStat::setValue, mIndex, prop }] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setValue(int index, std::string_view prop, const MWWorld::Ptr& ptr, const sol::object& value)
|
|
|
|
{
|
|
|
|
auto& stats = ptr.getClass().getCreatureStats(ptr);
|
|
|
|
auto stat = stats.getAttribute(index);
|
|
|
|
float floatValue = value.as<float>();
|
|
|
|
if (prop == "base")
|
|
|
|
stat.setBase(floatValue);
|
|
|
|
else if (prop == "damage")
|
|
|
|
{
|
|
|
|
stat.restore(stat.getDamage());
|
|
|
|
stat.damage(floatValue);
|
|
|
|
}
|
|
|
|
else if (prop == "modifier")
|
|
|
|
stat.setModifier(floatValue);
|
|
|
|
stats.setAttribute(index, stat);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class SkillStat
|
|
|
|
{
|
|
|
|
StatObject mObject;
|
|
|
|
int mIndex;
|
|
|
|
|
|
|
|
SkillStat(StatObject object, int index)
|
|
|
|
: mObject(std::move(object))
|
|
|
|
, mIndex(index)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static float getProgress(const MWWorld::Ptr& ptr, int index, const MWMechanics::SkillValue& stat)
|
|
|
|
{
|
|
|
|
float progress = stat.getProgress();
|
|
|
|
if (progress != 0.f)
|
|
|
|
progress /= getMaxProgress(ptr, index, stat);
|
|
|
|
return progress;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float getMaxProgress(const MWWorld::Ptr& ptr, int index, const MWMechanics::SkillValue& stat)
|
|
|
|
{
|
|
|
|
const auto& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
const auto cl = store.get<ESM::Class>().find(ptr.get<ESM::NPC>()->mBase->mClass);
|
|
|
|
return ptr.getClass().getNpcStats(ptr).getSkillProgressRequirement(index, *cl);
|
|
|
|
}
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
public:
|
|
|
|
template <class G>
|
|
|
|
sol::object get(const Context& context, std::string_view prop, G getter) const
|
|
|
|
{
|
2022-06-12 08:00:11 +00:00
|
|
|
return getValue(
|
|
|
|
context, mObject, &SkillStat::setValue, mIndex, prop, [this, getter](const MWLua::Object* obj) {
|
2022-03-25 20:03:13 +00:00
|
|
|
const auto& ptr = obj->ptr();
|
|
|
|
return (ptr.getClass().getNpcStats(ptr).getSkill(mIndex).*getter)();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-07-11 16:40:06 +02:00
|
|
|
float getModified(const Context& context) const
|
|
|
|
{
|
|
|
|
auto base = get(context, "base", &MWMechanics::SkillValue::getBase).as<float>();
|
|
|
|
auto damage = get(context, "damage", &MWMechanics::SkillValue::getDamage).as<float>();
|
|
|
|
auto modifier = get(context, "modifier", &MWMechanics::SkillValue::getModifier).as<float>();
|
|
|
|
return std::max(0.f, base - damage + modifier); // Should match SkillValue::getModified
|
|
|
|
}
|
|
|
|
|
2022-03-25 20:03:13 +00:00
|
|
|
sol::object getProgress(const Context& context) const
|
|
|
|
{
|
2022-06-12 08:00:11 +00:00
|
|
|
return getValue(
|
|
|
|
context, mObject, &SkillStat::setValue, mIndex, "progress", [this](const MWLua::Object* obj) {
|
2022-03-25 20:03:13 +00:00
|
|
|
const auto& ptr = obj->ptr();
|
|
|
|
return getProgress(ptr, mIndex, ptr.getClass().getNpcStats(ptr).getSkill(mIndex));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::optional<SkillStat> create(StatObject object, int index)
|
|
|
|
{
|
|
|
|
if (!getObject(object)->ptr().getClass().isNpc())
|
|
|
|
return {};
|
|
|
|
return SkillStat{ std::move(object), index };
|
|
|
|
}
|
|
|
|
|
|
|
|
void cache(const Context& context, std::string_view prop, const sol::object& value) const
|
|
|
|
{
|
2022-10-10 22:40:33 +02:00
|
|
|
SelfObject* obj = asSelfObject(mObject);
|
2022-03-25 20:03:13 +00:00
|
|
|
if (obj->mStatsCache.empty())
|
|
|
|
context.mLuaManager->addAction(std::make_unique<StatUpdateAction>(context.mLua, obj->id()));
|
|
|
|
obj->mStatsCache[SelfObject::CachedStat{ &SkillStat::setValue, mIndex, prop }] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setValue(int index, std::string_view prop, const MWWorld::Ptr& ptr, const sol::object& value)
|
|
|
|
{
|
|
|
|
auto& stats = ptr.getClass().getNpcStats(ptr);
|
|
|
|
auto stat = stats.getSkill(index);
|
|
|
|
float floatValue = value.as<float>();
|
|
|
|
if (prop == "base")
|
|
|
|
stat.setBase(floatValue);
|
|
|
|
else if (prop == "damage")
|
|
|
|
{
|
|
|
|
stat.restore(stat.getDamage());
|
|
|
|
stat.damage(floatValue);
|
|
|
|
}
|
|
|
|
else if (prop == "modifier")
|
|
|
|
stat.setModifier(floatValue);
|
|
|
|
else if (prop == "progress")
|
|
|
|
stat.setProgress(floatValue * getMaxProgress(ptr, index, stat));
|
|
|
|
stats.setSkill(index, stat);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace sol
|
|
|
|
{
|
|
|
|
template <>
|
|
|
|
struct is_automagical<MWLua::LevelStat> : std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
template <>
|
|
|
|
struct is_automagical<MWLua::DynamicStat> : std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
template <>
|
|
|
|
struct is_automagical<MWLua::AttributeStat> : std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
template <>
|
|
|
|
struct is_automagical<MWLua::SkillStat> : std::false_type
|
|
|
|
{
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace MWLua
|
|
|
|
{
|
|
|
|
void addActorStatsBindings(sol::table& actor, const Context& context)
|
|
|
|
{
|
|
|
|
sol::table stats(context.mLua->sol(), sol::create);
|
|
|
|
actor["stats"] = LuaUtil::makeReadOnly(stats);
|
|
|
|
|
|
|
|
auto levelStatT = context.mLua->sol().new_usertype<LevelStat>("LevelStat");
|
|
|
|
levelStatT["current"] = sol::property([context](const LevelStat& stat) { return stat.getCurrent(context); },
|
|
|
|
[context](const LevelStat& stat, const sol::object& value) { stat.setCurrent(context, value); });
|
|
|
|
levelStatT["progress"] = sol::property([context](const LevelStat& stat) { return stat.getProgress(context); });
|
|
|
|
stats["level"] = addIndexedAccessor<LevelStat>(0);
|
|
|
|
|
|
|
|
auto dynamicStatT = context.mLua->sol().new_usertype<DynamicStat>("DynamicStat");
|
|
|
|
addProp(context, dynamicStatT, "base", &MWMechanics::DynamicStat<float>::getBase);
|
|
|
|
addProp(context, dynamicStatT, "current", &MWMechanics::DynamicStat<float>::getCurrent);
|
|
|
|
addProp(context, dynamicStatT, "modifier", &MWMechanics::DynamicStat<float>::getModifier);
|
|
|
|
sol::table dynamic(context.mLua->sol(), sol::create);
|
|
|
|
stats["dynamic"] = LuaUtil::makeReadOnly(dynamic);
|
|
|
|
dynamic["health"] = addIndexedAccessor<DynamicStat>(0);
|
|
|
|
dynamic["magicka"] = addIndexedAccessor<DynamicStat>(1);
|
|
|
|
dynamic["fatigue"] = addIndexedAccessor<DynamicStat>(2);
|
|
|
|
|
|
|
|
auto attributeStatT = context.mLua->sol().new_usertype<AttributeStat>("AttributeStat");
|
|
|
|
addProp(context, attributeStatT, "base", &MWMechanics::AttributeValue::getBase);
|
|
|
|
addProp(context, attributeStatT, "damage", &MWMechanics::AttributeValue::getDamage);
|
2022-07-11 16:40:06 +02:00
|
|
|
attributeStatT["modified"]
|
|
|
|
= sol::property([=](const AttributeStat& stat) { return stat.getModified(context); });
|
2022-03-25 20:03:13 +00:00
|
|
|
addProp(context, attributeStatT, "modifier", &MWMechanics::AttributeValue::getModifier);
|
|
|
|
sol::table attributes(context.mLua->sol(), sol::create);
|
|
|
|
stats["attributes"] = LuaUtil::makeReadOnly(attributes);
|
|
|
|
for (int id = ESM::Attribute::Strength; id < ESM::Attribute::Length; ++id)
|
|
|
|
attributes[Misc::StringUtils::lowerCase(ESM::Attribute::sAttributeNames[id])]
|
|
|
|
= addIndexedAccessor<AttributeStat>(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void addNpcStatsBindings(sol::table& npc, const Context& context)
|
|
|
|
{
|
|
|
|
sol::table npcStats(context.mLua->sol(), sol::create);
|
|
|
|
sol::table baseMeta(context.mLua->sol(), sol::create);
|
|
|
|
baseMeta[sol::meta_function::index] = LuaUtil::getMutableFromReadOnly(npc["baseType"]["stats"]);
|
|
|
|
npcStats[sol::metatable_key] = baseMeta;
|
|
|
|
npc["stats"] = LuaUtil::makeReadOnly(npcStats);
|
|
|
|
|
|
|
|
auto skillStatT = context.mLua->sol().new_usertype<SkillStat>("SkillStat");
|
|
|
|
addProp(context, skillStatT, "base", &MWMechanics::SkillValue::getBase);
|
|
|
|
addProp(context, skillStatT, "damage", &MWMechanics::SkillValue::getDamage);
|
2022-07-11 16:40:06 +02:00
|
|
|
skillStatT["modified"] = sol::property([=](const SkillStat& stat) { return stat.getModified(context); });
|
2022-03-25 20:03:13 +00:00
|
|
|
addProp(context, skillStatT, "modifier", &MWMechanics::SkillValue::getModifier);
|
|
|
|
skillStatT["progress"] = sol::property([context](const SkillStat& stat) { return stat.getProgress(context); },
|
|
|
|
[context](const SkillStat& stat, const sol::object& value) { stat.cache(context, "progress", value); });
|
|
|
|
sol::table skills(context.mLua->sol(), sol::create);
|
|
|
|
npcStats["skills"] = LuaUtil::makeReadOnly(skills);
|
|
|
|
for (int id = ESM::Skill::Block; id < ESM::Skill::Length; ++id)
|
|
|
|
skills[Misc::StringUtils::lowerCase(ESM::Skill::sSkillNames[id])] = addIndexedAccessor<SkillStat>(id);
|
|
|
|
}
|
|
|
|
}
|