1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 11:23:52 +00:00
openmw/components/esm3/variant.cpp

279 lines
6.9 KiB
C++
Raw Normal View History

2013-03-04 12:59:06 +00:00
#include "variant.hpp"
#include <stdexcept>
#include "esmreader.hpp"
#include "variantimp.hpp"
namespace ESM
{
2022-09-22 18:26:05 +00:00
namespace
2013-03-04 12:59:06 +00:00
{
2022-09-22 18:26:05 +00:00
constexpr uint32_t STRV = fourCC("STRV");
constexpr uint32_t INTV = fourCC("INTV");
constexpr uint32_t FLTV = fourCC("FLTV");
constexpr uint32_t STTV = fourCC("STTV");
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
template <typename T, bool orDefault = false>
struct GetValue
{
2023-10-24 17:25:52 +00:00
constexpr T operator()(int32_t value) const { return static_cast<T>(value); }
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
constexpr T operator()(float value) const { return static_cast<T>(value); }
2022-09-22 18:26:05 +00:00
template <typename V>
constexpr T operator()(const V&) const
{
if constexpr (orDefault)
return T{};
else
throw std::runtime_error("cannot convert variant");
}
};
2022-09-22 18:26:05 +00:00
template <typename T>
struct SetValue
{
T mValue;
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
explicit SetValue(T value)
: mValue(value)
{
}
2023-10-24 17:25:52 +00:00
void operator()(int32_t& value) const { value = static_cast<int32_t>(mValue); }
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
void operator()(float& value) const { value = static_cast<float>(mValue); }
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
template <typename V>
void operator()(V&) const
{
throw std::runtime_error("cannot convert variant");
}
};
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
const std::string& Variant::getString() const
{
return std::get<std::string>(mData);
}
2013-03-04 12:59:06 +00:00
2023-10-24 17:25:52 +00:00
int32_t Variant::getInteger() const
2022-09-22 18:26:05 +00:00
{
2023-10-24 17:25:52 +00:00
return std::visit(GetValue<int32_t>{}, mData);
2022-09-22 18:26:05 +00:00
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
float Variant::getFloat() const
2013-03-04 12:59:06 +00:00
{
2022-09-22 18:26:05 +00:00
return std::visit(GetValue<float>{}, mData);
2013-03-04 12:59:06 +00:00
}
2022-09-22 18:26:05 +00:00
void Variant::read(ESMReader& esm, Format format)
2013-03-04 12:59:06 +00:00
{
2022-09-22 18:26:05 +00:00
// type
VarType type = VT_Unknown;
if (format == Format_Global)
2013-03-04 12:59:06 +00:00
{
2022-09-22 18:26:05 +00:00
std::string typeId = esm.getHNString("FNAM");
if (typeId == "s")
type = VT_Short;
else if (typeId == "l")
type = VT_Long;
else if (typeId == "f")
type = VT_Float;
else
esm.fail("illegal global variable type " + typeId);
2013-03-04 12:59:06 +00:00
}
2022-09-22 18:26:05 +00:00
else if (format == Format_Gmst)
{
if (!esm.hasMoreSubs())
{
type = VT_None;
}
else
{
esm.getSubName();
NAME name = esm.retSubName();
if (name == STRV)
{
type = VT_String;
}
else if (name == INTV)
{
type = VT_Int;
}
else if (name == FLTV)
{
type = VT_Float;
}
else
esm.fail("invalid subrecord: " + name.toString());
}
}
else if (format == Format_Info)
2013-03-04 12:59:06 +00:00
{
2013-03-05 07:02:05 +00:00
esm.getSubName();
NAME name = esm.retSubName();
2022-09-22 18:26:05 +00:00
if (name == INTV)
2013-03-05 07:02:05 +00:00
{
2022-09-22 18:26:05 +00:00
type = VT_Int;
}
else if (name == FLTV)
{
type = VT_Float;
2013-03-05 07:02:05 +00:00
}
2022-09-22 18:26:05 +00:00
else
esm.fail("invalid subrecord: " + name.toString());
}
else if (format == Format_Local)
{
esm.getSubName();
NAME name = esm.retSubName();
if (name == INTV)
2013-03-05 07:02:05 +00:00
{
type = VT_Int;
}
2022-09-22 18:26:05 +00:00
else if (name == FLTV)
2013-03-05 07:02:05 +00:00
{
type = VT_Float;
}
2022-09-22 18:26:05 +00:00
else if (name == STTV)
{
type = VT_Short;
}
2013-03-05 07:02:05 +00:00
else
2022-09-22 18:26:05 +00:00
esm.fail("invalid subrecord: " + name.toString());
2013-03-04 12:59:06 +00:00
}
2022-09-22 18:26:05 +00:00
setType(type);
std::visit(ReadESMVariantValue{ esm, format, mType }, mData);
}
2022-09-22 18:26:05 +00:00
void Variant::write(ESMWriter& esm, Format format) const
{
if (mType == VT_Unknown)
{
2022-09-22 18:26:05 +00:00
throw std::runtime_error("can not serialise variant of unknown type");
}
2022-09-22 18:26:05 +00:00
else if (mType == VT_None)
{
2022-09-22 18:26:05 +00:00
if (format == Format_Global)
throw std::runtime_error("can not serialise variant of type none to global format");
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
if (format == Format_Info)
throw std::runtime_error("can not serialise variant of type none to info format");
2022-09-22 18:26:05 +00:00
if (format == Format_Local)
throw std::runtime_error("can not serialise variant of type none to local format");
2022-09-22 18:26:05 +00:00
// nothing to do here for GMST format
}
else
std::visit(WriteESMVariantValue{ esm, format, mType }, mData);
2013-03-04 12:59:06 +00:00
}
2022-09-22 18:26:05 +00:00
void Variant::write(std::ostream& stream) const
2013-03-04 12:59:06 +00:00
{
2022-09-22 18:26:05 +00:00
switch (mType)
{
case VT_Unknown:
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
stream << "variant unknown";
break;
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
case VT_None:
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
stream << "variant none";
break;
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
case VT_Short:
2013-03-04 12:59:06 +00:00
2023-10-24 17:25:52 +00:00
stream << "variant short: " << std::get<int32_t>(mData);
2022-09-22 18:26:05 +00:00
break;
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
case VT_Int:
2013-03-04 12:59:06 +00:00
2023-10-24 17:25:52 +00:00
stream << "variant int: " << std::get<int32_t>(mData);
break;
2013-03-04 12:59:06 +00:00
case VT_Long:
2022-09-22 18:26:05 +00:00
2023-10-24 17:25:52 +00:00
stream << "variant long: " << std::get<int32_t>(mData);
2013-03-04 12:59:06 +00:00
break;
case VT_Float:
2022-09-22 18:26:05 +00:00
stream << "variant float: " << std::get<float>(mData);
2013-03-04 12:59:06 +00:00
break;
case VT_String:
2022-09-22 18:26:05 +00:00
stream << "variant string: \"" << std::get<std::string>(mData) << "\"";
2013-03-04 12:59:06 +00:00
break;
}
2022-09-22 18:26:05 +00:00
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
void Variant::setType(VarType type)
{
if (type != mType)
{
switch (type)
{
case VT_Unknown:
case VT_None:
mData = std::monostate{};
break;
case VT_Short:
case VT_Int:
case VT_Long:
mData = std::visit(GetValue<int, true>{}, mData);
break;
case VT_Float:
mData = std::visit(GetValue<float, true>{}, mData);
break;
case VT_String:
mData = std::string{};
break;
}
mType = type;
}
2013-03-04 12:59:06 +00:00
}
2022-09-22 18:26:05 +00:00
void Variant::setString(const std::string& value)
{
std::get<std::string>(mData) = value;
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
void Variant::setString(std::string&& value)
{
std::get<std::string>(mData) = std::move(value);
}
2013-03-04 12:59:06 +00:00
2023-10-24 17:25:52 +00:00
void Variant::setInteger(int32_t value)
2022-09-22 18:26:05 +00:00
{
std::visit(SetValue(value), mData);
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
void Variant::setFloat(float value)
{
std::visit(SetValue(value), mData);
}
2013-03-04 12:59:06 +00:00
2022-09-22 18:26:05 +00:00
std::ostream& operator<<(std::ostream& stream, const Variant& value)
{
value.write(stream);
return stream;
}
}