#include "variantimp.hpp" #include <stdexcept> #include "esmreader.hpp" #include "esmwriter.hpp" ESM::VariantDataBase::~VariantDataBase() {} std::string ESM::VariantDataBase::getString (bool default_) const { if (default_) return ""; throw std::runtime_error ("can not convert variant to string"); } int ESM::VariantDataBase::getInteger (bool default_) const { if (default_) return 0; throw std::runtime_error ("can not convert variant to integer"); } float ESM::VariantDataBase::getFloat (bool default_) const { if (default_) return 0; throw std::runtime_error ("can not convert variant to float"); } void ESM::VariantDataBase::setString (const std::string& value) { throw std::runtime_error ("conversion of string to variant not possible"); } void ESM::VariantDataBase::setInteger (int value) { throw std::runtime_error ("conversion of integer to variant not possible"); } void ESM::VariantDataBase::setFloat (float value) { throw std::runtime_error ("conversion of float to variant not possible"); } ESM::VariantStringData::VariantStringData (const VariantDataBase *data) { if (data) mValue = data->getString (true); } ESM::VariantDataBase *ESM::VariantStringData::clone() const { return new VariantStringData (*this); } std::string ESM::VariantStringData::getString (bool default_) const { return mValue; } void ESM::VariantStringData::setString (const std::string& value) { mValue = value; } void ESM::VariantStringData::read (ESMReader& esm, Variant::Format format, VarType type) { if (type!=VT_String) throw std::logic_error ("not a string type"); if (format==Variant::Format_Global) esm.fail ("global variables of type string not supported"); if (format==Variant::Format_Info) esm.fail ("info variables of type string not supported"); if (format==Variant::Format_Local) esm.fail ("local variables of type string not supported"); // GMST mValue = esm.getHString(); } void ESM::VariantStringData::write (ESMWriter& esm, Variant::Format format, VarType type) const { if (type!=VT_String) throw std::logic_error ("not a string type"); if (format==Variant::Format_Global) throw std::runtime_error ("global variables of type string not supported"); if (format==Variant::Format_Info) throw std::runtime_error ("info variables of type string not supported"); // GMST esm.writeHNString ("STRV", mValue); } bool ESM::VariantStringData::isEqual (const VariantDataBase& value) const { return dynamic_cast<const VariantStringData&> (value).mValue==mValue; } ESM::VariantIntegerData::VariantIntegerData (const VariantDataBase *data) : mValue (0) { if (data) mValue = data->getInteger (true); } ESM::VariantDataBase *ESM::VariantIntegerData::clone() const { return new VariantIntegerData (*this); } int ESM::VariantIntegerData::getInteger (bool default_) const { return mValue; } float ESM::VariantIntegerData::getFloat (bool default_) const { return static_cast<float>(mValue); } void ESM::VariantIntegerData::setInteger (int value) { mValue = value; } void ESM::VariantIntegerData::setFloat (float value) { mValue = static_cast<int> (value); } void ESM::VariantIntegerData::read (ESMReader& esm, Variant::Format format, VarType type) { if (type!=VT_Short && type!=VT_Long && type!=VT_Int) throw std::logic_error ("not an integer type"); if (format==Variant::Format_Global) { float value; esm.getHNT (value, "FLTV"); if (type==VT_Short) { if (value!=value) mValue = 0; // nan else mValue = static_cast<short> (value); } else if (type==VT_Long) mValue = static_cast<int> (value); else esm.fail ("unsupported global variable integer type"); } else if (format==Variant::Format_Gmst || format==Variant::Format_Info) { if (type!=VT_Int) { std::ostringstream stream; stream << "unsupported " <<(format==Variant::Format_Gmst ? "gmst" : "info") << " variable integer type"; esm.fail (stream.str()); } esm.getHT (mValue); } else if (format==Variant::Format_Local) { if (type==VT_Short) { short value; esm.getHT(value); mValue = value; } else if (type==VT_Int) { esm.getHT(mValue); } else esm.fail("unsupported local variable integer type"); } } void ESM::VariantIntegerData::write (ESMWriter& esm, Variant::Format format, VarType type) const { if (type!=VT_Short && type!=VT_Long && type!=VT_Int) throw std::logic_error ("not an integer type"); if (format==Variant::Format_Global) { if (type==VT_Short || type==VT_Long) { float value = static_cast<float>(mValue); esm.writeHNString ("FNAM", type==VT_Short ? "s" : "l"); esm.writeHNT ("FLTV", value); } else throw std::runtime_error ("unsupported global variable integer type"); } else if (format==Variant::Format_Gmst || format==Variant::Format_Info) { if (type!=VT_Int) { std::ostringstream stream; stream << "unsupported " <<(format==Variant::Format_Gmst ? "gmst" : "info") << " variable integer type"; throw std::runtime_error (stream.str()); } esm.writeHNT ("INTV", mValue); } else if (format==Variant::Format_Local) { if (type==VT_Short) esm.writeHNT ("STTV", (short)mValue); else if (type == VT_Int) esm.writeHNT ("INTV", mValue); else throw std::runtime_error("unsupported local variable integer type"); } } bool ESM::VariantIntegerData::isEqual (const VariantDataBase& value) const { return dynamic_cast<const VariantIntegerData&> (value).mValue==mValue; } ESM::VariantFloatData::VariantFloatData (const VariantDataBase *data) : mValue (0) { if (data) mValue = data->getFloat (true); } ESM::VariantDataBase *ESM::VariantFloatData::clone() const { return new VariantFloatData (*this); } int ESM::VariantFloatData::getInteger (bool default_) const { return static_cast<int> (mValue); } float ESM::VariantFloatData::getFloat (bool default_) const { return mValue; } void ESM::VariantFloatData::setInteger (int value) { mValue = static_cast<float>(value); } void ESM::VariantFloatData::setFloat (float value) { mValue = value; } void ESM::VariantFloatData::read (ESMReader& esm, Variant::Format format, VarType type) { if (type!=VT_Float) throw std::logic_error ("not a float type"); if (format==Variant::Format_Global) { esm.getHNT (mValue, "FLTV"); } else if (format==Variant::Format_Gmst || format==Variant::Format_Info || format==Variant::Format_Local) { esm.getHT (mValue); } } void ESM::VariantFloatData::write (ESMWriter& esm, Variant::Format format, VarType type) const { if (type!=VT_Float) throw std::logic_error ("not a float type"); if (format==Variant::Format_Global) { esm.writeHNString ("FNAM", "f"); esm.writeHNT ("FLTV", mValue); } else if (format==Variant::Format_Gmst || format==Variant::Format_Info || format==Variant::Format_Local) { esm.writeHNT ("FLTV", mValue); } } bool ESM::VariantFloatData::isEqual (const VariantDataBase& value) const { return dynamic_cast<const VariantFloatData&> (value).mValue==mValue; }