2012-11-24 10:01:53 +00:00
|
|
|
|
|
|
|
#include "universalid.hpp"
|
|
|
|
|
|
|
|
#include <ostream>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
struct TypeData
|
|
|
|
{
|
|
|
|
CSMWorld::UniversalId::Class mClass;
|
|
|
|
CSMWorld::UniversalId::Type mType;
|
|
|
|
const char *mName;
|
|
|
|
};
|
|
|
|
|
|
|
|
static const TypeData sNoArg[] =
|
|
|
|
{
|
|
|
|
{ CSMWorld::UniversalId::Class_None, CSMWorld::UniversalId::Type_None, "empty" },
|
2012-11-26 11:29:22 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Globals, "Global Variables" },
|
2013-02-08 08:58:19 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Gmsts, "Game Settings" },
|
2013-03-21 12:40:55 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Skills, "Skills" },
|
2013-03-25 12:22:06 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Classes, "Classes" },
|
2013-04-02 10:00:45 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Factions, "Factions" },
|
2013-04-04 12:34:39 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Races, "Races" },
|
2013-04-06 19:21:10 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Sounds, "Sounds" },
|
2013-04-07 13:17:35 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Scripts, "Scripts" },
|
2013-04-07 14:32:06 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_RecordList, CSMWorld::UniversalId::Type_Regions, "Regions" },
|
2012-11-24 10:01:53 +00:00
|
|
|
|
|
|
|
{ CSMWorld::UniversalId::Class_None, CSMWorld::UniversalId::Type_None, 0 } // end marker
|
|
|
|
};
|
|
|
|
|
|
|
|
static const TypeData sIdArg[] =
|
|
|
|
{
|
2012-12-13 12:35:08 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Global, "Global Variable" },
|
2013-02-08 08:58:19 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Gmst, "Game Setting" },
|
2013-03-21 12:40:55 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Skill, "Skill" },
|
2013-03-25 12:22:06 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Class, "Class" },
|
2013-04-02 10:00:45 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Faction, "Faction" },
|
2013-04-04 12:34:39 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Race, "Race" },
|
2013-04-06 19:21:10 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Sound, "Sound" },
|
2013-04-07 13:17:35 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Script, "Script" },
|
2013-04-07 14:32:06 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Record, CSMWorld::UniversalId::Type_Region, "Region" },
|
2012-11-24 10:01:53 +00:00
|
|
|
|
|
|
|
{ CSMWorld::UniversalId::Class_None, CSMWorld::UniversalId::Type_None, 0 } // end marker
|
|
|
|
};
|
|
|
|
|
|
|
|
static const TypeData sIndexArg[] =
|
|
|
|
{
|
2012-12-11 14:35:47 +00:00
|
|
|
{ CSMWorld::UniversalId::Class_Transient, CSMWorld::UniversalId::Type_VerificationResults, "Verification Results" },
|
2012-11-24 10:01:53 +00:00
|
|
|
|
|
|
|
{ CSMWorld::UniversalId::Class_None, CSMWorld::UniversalId::Type_None, 0 } // end marker
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2012-12-08 22:27:59 +00:00
|
|
|
CSMWorld::UniversalId::UniversalId (const std::string& universalId)
|
|
|
|
{
|
|
|
|
std::string::size_type index = universalId.find (':');
|
|
|
|
|
2013-03-25 10:04:41 +00:00
|
|
|
if (index!=std::string::npos)
|
2012-12-08 22:27:59 +00:00
|
|
|
{
|
|
|
|
std::string type = universalId.substr (0, index);
|
|
|
|
|
2013-04-06 17:20:46 +00:00
|
|
|
for (int i=0; sIdArg[i].mName; ++i)
|
|
|
|
if (type==sIdArg[i].mName)
|
|
|
|
{
|
|
|
|
mArgumentType = ArgumentType_Id;
|
|
|
|
mType = sIdArg[i].mType;
|
|
|
|
mClass = sIdArg[i].mClass;
|
|
|
|
mId = universalId.substr (index+2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i=0; sIndexArg[i].mName; ++i)
|
|
|
|
if (type==sIndexArg[i].mName)
|
|
|
|
{
|
|
|
|
mArgumentType = ArgumentType_Index;
|
|
|
|
mType = sIndexArg[i].mType;
|
|
|
|
mClass = sIndexArg[i].mClass;
|
|
|
|
|
|
|
|
std::istringstream stream (universalId.substr (index+2));
|
|
|
|
|
|
|
|
if (stream >> mIndex)
|
2012-12-09 10:10:35 +00:00
|
|
|
return;
|
|
|
|
|
2013-04-06 17:20:46 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int i=0; sNoArg[i].mName; ++i)
|
|
|
|
if (universalId==sNoArg[i].mName)
|
|
|
|
{
|
|
|
|
mArgumentType = ArgumentType_None;
|
|
|
|
mType = sNoArg[i].mType;
|
|
|
|
mClass = sNoArg[i].mClass;
|
|
|
|
return;
|
|
|
|
}
|
2012-12-08 22:27:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
throw std::runtime_error ("invalid UniversalId: " + universalId);
|
|
|
|
}
|
|
|
|
|
2012-11-24 10:01:53 +00:00
|
|
|
CSMWorld::UniversalId::UniversalId (Type type) : mArgumentType (ArgumentType_None), mType (type), mIndex (0)
|
|
|
|
{
|
|
|
|
for (int i=0; sNoArg[i].mName; ++i)
|
|
|
|
if (type==sNoArg[i].mType)
|
|
|
|
{
|
|
|
|
mClass = sNoArg[i].mClass;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw std::logic_error ("invalid argument-less UniversalId type");
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::UniversalId::UniversalId (Type type, const std::string& id)
|
|
|
|
: mArgumentType (ArgumentType_Id), mType (type), mId (id), mIndex (0)
|
|
|
|
{
|
|
|
|
for (int i=0; sIdArg[i].mName; ++i)
|
|
|
|
if (type==sIdArg[i].mType)
|
|
|
|
{
|
|
|
|
mClass = sIdArg[i].mClass;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw std::logic_error ("invalid ID argument UniversalId type");
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::UniversalId::UniversalId (Type type, int index)
|
|
|
|
: mArgumentType (ArgumentType_Index), mType (type), mIndex (index)
|
|
|
|
{
|
|
|
|
for (int i=0; sIndexArg[i].mName; ++i)
|
|
|
|
if (type==sIndexArg[i].mType)
|
|
|
|
{
|
|
|
|
mClass = sIndexArg[i].mClass;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw std::logic_error ("invalid index argument UniversalId type");
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::UniversalId::Class CSMWorld::UniversalId::getClass() const
|
|
|
|
{
|
|
|
|
return mClass;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::UniversalId::ArgumentType CSMWorld::UniversalId::getArgumentType() const
|
|
|
|
{
|
|
|
|
return mArgumentType;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::UniversalId::Type CSMWorld::UniversalId::getType() const
|
|
|
|
{
|
|
|
|
return mType;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& CSMWorld::UniversalId::getId() const
|
|
|
|
{
|
|
|
|
if (mArgumentType!=ArgumentType_Id)
|
|
|
|
throw std::logic_error ("invalid access to ID of non-ID UniversalId");
|
|
|
|
|
|
|
|
return mId;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::UniversalId::getIndex() const
|
|
|
|
{
|
|
|
|
if (mArgumentType!=ArgumentType_Index)
|
|
|
|
throw std::logic_error ("invalid access to index of non-index UniversalId");
|
|
|
|
|
|
|
|
return mIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSMWorld::UniversalId::isEqual (const UniversalId& universalId) const
|
|
|
|
{
|
|
|
|
if (mClass!=universalId.mClass || mArgumentType!=universalId.mArgumentType || mType!=universalId.mType)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (mArgumentType)
|
|
|
|
{
|
|
|
|
case ArgumentType_Id: return mId==universalId.mId;
|
|
|
|
case ArgumentType_Index: return mIndex==universalId.mIndex;
|
|
|
|
|
|
|
|
default: return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-26 11:29:22 +00:00
|
|
|
bool CSMWorld::UniversalId::isLess (const UniversalId& universalId) const
|
|
|
|
{
|
|
|
|
if (mType<universalId.mType)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (mType>universalId.mType)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (mArgumentType)
|
|
|
|
{
|
|
|
|
case ArgumentType_Id: return mId<universalId.mId;
|
|
|
|
case ArgumentType_Index: return mIndex<universalId.mIndex;
|
|
|
|
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-24 10:01:53 +00:00
|
|
|
std::string CSMWorld::UniversalId::getTypeName() const
|
|
|
|
{
|
|
|
|
const TypeData *typeData = mArgumentType==ArgumentType_None ? sNoArg :
|
|
|
|
(mArgumentType==ArgumentType_Id ? sIdArg : sIndexArg);
|
|
|
|
|
|
|
|
for (int i=0; typeData[i].mName; ++i)
|
|
|
|
if (typeData[i].mType==mType)
|
|
|
|
return typeData[i].mName;
|
|
|
|
|
|
|
|
throw std::logic_error ("failed to retrieve UniversalId type name");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string CSMWorld::UniversalId::toString() const
|
|
|
|
{
|
|
|
|
std::ostringstream stream;
|
|
|
|
|
2012-12-09 10:10:35 +00:00
|
|
|
stream << getTypeName();
|
2012-11-24 10:01:53 +00:00
|
|
|
|
|
|
|
switch (mArgumentType)
|
|
|
|
{
|
|
|
|
case ArgumentType_None: break;
|
2012-12-13 12:35:08 +00:00
|
|
|
case ArgumentType_Id: stream << ": " << mId; break;
|
|
|
|
case ArgumentType_Index: stream << ": " << mIndex; break;
|
2012-11-24 10:01:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return stream.str();
|
|
|
|
}
|
|
|
|
|
2012-11-26 11:29:22 +00:00
|
|
|
bool CSMWorld::operator== (const CSMWorld::UniversalId& left, const CSMWorld::UniversalId& right)
|
2012-11-24 10:01:53 +00:00
|
|
|
{
|
|
|
|
return left.isEqual (right);
|
|
|
|
}
|
|
|
|
|
2012-11-26 11:29:22 +00:00
|
|
|
bool CSMWorld::operator!= (const CSMWorld::UniversalId& left, const CSMWorld::UniversalId& right)
|
2012-11-24 10:01:53 +00:00
|
|
|
{
|
|
|
|
return !left.isEqual (right);
|
|
|
|
}
|
|
|
|
|
2012-11-26 11:29:22 +00:00
|
|
|
bool CSMWorld::operator< (const UniversalId& left, const UniversalId& right)
|
|
|
|
{
|
|
|
|
return left.isLess (right);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& CSMWorld::operator< (std::ostream& stream, const CSMWorld::UniversalId& universalId)
|
2012-11-24 10:01:53 +00:00
|
|
|
{
|
|
|
|
return stream << universalId.toString();
|
|
|
|
}
|