1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-25 22:26:37 +00:00

ESMStore: no longer necessary to include the type definitions in the header

For now there still needs to declare all the getters, but a macro may make that easier, or an different method entierly
This commit is contained in:
florent.teppe 2022-07-03 11:47:46 +02:00
parent 5815faecda
commit a4c1bff03d
2 changed files with 451 additions and 302 deletions

View file

@ -130,6 +130,109 @@ namespace
namespace MWWorld namespace MWWorld
{ {
struct ESMStoreImp
{
Store<ESM::Activator> mActivators;
Store<ESM::Potion> mPotions;
Store<ESM::Apparatus> mAppas;
Store<ESM::Armor> mArmors;
Store<ESM::BodyPart> mBodyParts;
Store<ESM::Book> mBooks;
Store<ESM::BirthSign> mBirthSigns;
Store<ESM::Class> mClasses;
Store<ESM::Clothing> mClothes;
Store<ESM::Container> mContainers;
Store<ESM::Creature> mCreatures;
Store<ESM::Dialogue> mDialogs;
Store<ESM::Door> mDoors;
Store<ESM::Enchantment> mEnchants;
Store<ESM::Faction> mFactions;
Store<ESM::Global> mGlobals;
Store<ESM::Ingredient> mIngreds;
Store<ESM::CreatureLevList> mCreatureLists;
Store<ESM::ItemLevList> mItemLists;
Store<ESM::Light> mLights;
Store<ESM::Lockpick> mLockpicks;
Store<ESM::Miscellaneous> mMiscItems;
Store<ESM::NPC> mNpcs;
Store<ESM::Probe> mProbes;
Store<ESM::Race> mRaces;
Store<ESM::Region> mRegions;
Store<ESM::Repair> mRepairs;
Store<ESM::SoundGenerator> mSoundGens;
Store<ESM::Sound> mSounds;
Store<ESM::Spell> mSpells;
Store<ESM::StartScript> mStartScripts;
Store<ESM::Static> mStatics;
Store<ESM::Weapon> mWeapons;
Store<ESM::GameSetting> mGameSettings;
Store<ESM::Script> mScripts;
// Lists that need special rules
Store<ESM::Cell> mCells;
Store<ESM::Land> mLands;
Store<ESM::LandTexture> mLandTextures;
Store<ESM::Pathgrid> mPathgrids;
Store<ESM::MagicEffect> mMagicEffects;
Store<ESM::Skill> mSkills;
// Special entry which is hardcoded and not loaded from an ESM
Store<ESM::Attribute> mAttributes;
};
ESMStore::ESMStore()
{
mStoreImp = std::make_unique<ESMStoreImp>();
mDynamicCount = 0;
mStores[ESM::REC_ACTI] = &mStoreImp->mActivators;
mStores[ESM::REC_ALCH] = &mStoreImp->mPotions;
mStores[ESM::REC_APPA] = &mStoreImp->mAppas;
mStores[ESM::REC_ARMO] = &mStoreImp->mArmors;
mStores[ESM::REC_BODY] = &mStoreImp->mBodyParts;
mStores[ESM::REC_BOOK] = &mStoreImp->mBooks;
mStores[ESM::REC_BSGN] = &mStoreImp->mBirthSigns;
mStores[ESM::REC_CELL] = &mStoreImp->mCells;
mStores[ESM::REC_CLAS] = &mStoreImp->mClasses;
mStores[ESM::REC_CLOT] = &mStoreImp->mClothes;
mStores[ESM::REC_CONT] = &mStoreImp->mContainers;
mStores[ESM::REC_CREA] = &mStoreImp->mCreatures;
mStores[ESM::REC_DIAL] = &mStoreImp->mDialogs;
mStores[ESM::REC_DOOR] = &mStoreImp->mDoors;
mStores[ESM::REC_ENCH] = &mStoreImp->mEnchants;
mStores[ESM::REC_FACT] = &mStoreImp->mFactions;
mStores[ESM::REC_GLOB] = &mStoreImp->mGlobals;
mStores[ESM::REC_GMST] = &mStoreImp->mGameSettings;
mStores[ESM::REC_INGR] = &mStoreImp->mIngreds;
mStores[ESM::REC_LAND] = &mStoreImp->mLands;
mStores[ESM::REC_LEVC] = &mStoreImp->mCreatureLists;
mStores[ESM::REC_LEVI] = &mStoreImp->mItemLists;
mStores[ESM::REC_LIGH] = &mStoreImp->mLights;
mStores[ESM::REC_LOCK] = &mStoreImp->mLockpicks;
mStores[ESM::REC_LTEX] = &mStoreImp->mLandTextures;
mStores[ESM::REC_MISC] = &mStoreImp->mMiscItems;
mStores[ESM::REC_NPC_] = &mStoreImp->mNpcs;
mStores[ESM::REC_PGRD] = &mStoreImp->mPathgrids;
mStores[ESM::REC_PROB] = &mStoreImp->mProbes;
mStores[ESM::REC_RACE] = &mStoreImp->mRaces;
mStores[ESM::REC_REGN] = &mStoreImp->mRegions;
mStores[ESM::REC_REPA] = &mStoreImp->mRepairs;
mStores[ESM::REC_SCPT] = &mStoreImp->mScripts;
mStores[ESM::REC_SNDG] = &mStoreImp->mSoundGens;
mStores[ESM::REC_SOUN] = &mStoreImp->mSounds;
mStores[ESM::REC_SPEL] = &mStoreImp->mSpells;
mStores[ESM::REC_SSCR] = &mStoreImp->mStartScripts;
mStores[ESM::REC_STAT] = &mStoreImp->mStatics;
mStores[ESM::REC_WEAP] = &mStoreImp->mWeapons;
mStoreImp->mPathgrids.setCells(mStoreImp->mCells);
}
ESMStore::~ESMStore()
{
}
static bool isCacheableRecord(int id) static bool isCacheableRecord(int id)
{ {
if (id == ESM::REC_ACTI || id == ESM::REC_ALCH || id == ESM::REC_APPA || id == ESM::REC_ARMO || if (id == ESM::REC_ACTI || id == ESM::REC_ALCH || id == ESM::REC_APPA || id == ESM::REC_ARMO ||
@ -152,7 +255,7 @@ void ESMStore::load(ESM::ESMReader &esm, Loading::Listener* listener, ESM::Dialo
// Land texture loading needs to use a separate internal store for each plugin. // Land texture loading needs to use a separate internal store for each plugin.
// We set the number of plugins here so we can properly verify if valid plugin // We set the number of plugins here so we can properly verify if valid plugin
// indices are being passed to the LandTexture Store retrieval methods. // indices are being passed to the LandTexture Store retrieval methods.
mLandTextures.resize(esm.getIndex()+1); mStoreImp->mLandTextures.resize(esm.getIndex()+1);
// Loop through all records // Loop through all records
while(esm.hasMoreRecs()) while(esm.hasMoreRecs())
@ -180,9 +283,9 @@ void ESMStore::load(ESM::ESMReader &esm, Loading::Listener* listener, ESM::Dialo
esm.skipRecord(); esm.skipRecord();
} }
} else if (n.toInt() == ESM::REC_MGEF) { } else if (n.toInt() == ESM::REC_MGEF) {
mMagicEffects.load (esm); mStoreImp->mMagicEffects.load (esm);
} else if (n.toInt() == ESM::REC_SKIL) { } else if (n.toInt() == ESM::REC_SKIL) {
mSkills.load (esm); mStoreImp->mSkills.load (esm);
} }
else if (n.toInt() == ESM::REC_FILT || n.toInt() == ESM::REC_DBGP) else if (n.toInt() == ESM::REC_FILT || n.toInt() == ESM::REC_DBGP)
{ {
@ -208,7 +311,7 @@ void ESMStore::load(ESM::ESMReader &esm, Loading::Listener* listener, ESM::Dialo
} }
if (n.toInt() == ESM::REC_DIAL) { if (n.toInt() == ESM::REC_DIAL) {
dialogue = const_cast<ESM::Dialogue*>(mDialogs.find(id.mId)); dialogue = const_cast<ESM::Dialogue*>(mStoreImp->mDialogs.find(id.mId));
} else { } else {
dialogue = nullptr; dialogue = nullptr;
} }
@ -266,10 +369,10 @@ void ESMStore::setUp()
for (const auto& [k, v] : mIds) for (const auto& [k, v] : mIds)
mStaticIds.emplace(Misc::StringUtils::lowerCase(k), v); mStaticIds.emplace(Misc::StringUtils::lowerCase(k), v);
mSkills.setUp(); mStoreImp->mSkills.setUp();
mMagicEffects.setUp(); mStoreImp->mMagicEffects.setUp();
mAttributes.setUp(); mStoreImp->mAttributes.setUp();
mDialogs.setUp(); mStoreImp->mDialogs.setUp();
} }
void ESMStore::validateRecords(ESM::ReadersCache& readers) void ESMStore::validateRecords(ESM::ReadersCache& readers)
@ -286,9 +389,9 @@ void ESMStore::countAllCellRefs(ESM::ReadersCache& readers)
return; return;
std::vector<Ref> refs; std::vector<Ref> refs;
std::vector<std::string> refIDs; std::vector<std::string> refIDs;
for(auto it = mCells.intBegin(); it != mCells.intEnd(); ++it) for(auto it = mStoreImp->mCells.intBegin(); it != mStoreImp->mCells.intEnd(); ++it)
readRefs(*it, refs, refIDs, readers); readRefs(*it, refs, refIDs, readers);
for(auto it = mCells.extBegin(); it != mCells.extEnd(); ++it) for(auto it = mStoreImp->mCells.extBegin(); it != mStoreImp->mCells.extEnd(); ++it)
readRefs(*it, refs, refIDs, readers); readRefs(*it, refs, refIDs, readers);
const auto lessByRefNum = [] (const Ref& l, const Ref& r) { return l.mRefNum < r.mRefNum; }; const auto lessByRefNum = [] (const Ref& l, const Ref& r) { return l.mRefNum < r.mRefNum; };
std::stable_sort(refs.begin(), refs.end(), lessByRefNum); std::stable_sort(refs.begin(), refs.end(), lessByRefNum);
@ -317,17 +420,17 @@ int ESMStore::getRefCount(std::string_view id) const
void ESMStore::validate() void ESMStore::validate()
{ {
std::vector<ESM::NPC> npcsToReplace = getNPCsToReplace(mFactions, mClasses, mNpcs.mStatic); std::vector<ESM::NPC> npcsToReplace = getNPCsToReplace(mStoreImp->mFactions, mStoreImp->mClasses, mStoreImp->mNpcs.mStatic);
for (const ESM::NPC &npc : npcsToReplace) for (const ESM::NPC &npc : npcsToReplace)
{ {
mNpcs.eraseStatic(npc.mId); mStoreImp->mNpcs.eraseStatic(npc.mId);
mNpcs.insertStatic(npc); mStoreImp->mNpcs.insertStatic(npc);
} }
// Validate spell effects for invalid arguments // Validate spell effects for invalid arguments
std::vector<ESM::Spell> spellsToReplace; std::vector<ESM::Spell> spellsToReplace;
for (ESM::Spell spell : mSpells) for (ESM::Spell spell : mStoreImp->mSpells)
{ {
if (spell.mEffects.mList.empty()) if (spell.mEffects.mList.empty())
continue; continue;
@ -336,7 +439,7 @@ void ESMStore::validate()
auto iter = spell.mEffects.mList.begin(); auto iter = spell.mEffects.mList.begin();
while (iter != spell.mEffects.mList.end()) while (iter != spell.mEffects.mList.end())
{ {
const ESM::MagicEffect* mgef = mMagicEffects.search(iter->mEffectID); const ESM::MagicEffect* mgef = mStoreImp->mMagicEffects.search(iter->mEffectID);
if (!mgef) if (!mgef)
{ {
Log(Debug::Verbose) << "Spell '" << spell.mId << "' has an invalid effect (index " << iter->mEffectID << ") present. Dropping the effect."; Log(Debug::Verbose) << "Spell '" << spell.mId << "' has an invalid effect (index " << iter->mEffectID << ") present. Dropping the effect.";
@ -383,25 +486,31 @@ void ESMStore::validate()
for (const ESM::Spell &spell : spellsToReplace) for (const ESM::Spell &spell : spellsToReplace)
{ {
mSpells.eraseStatic(spell.mId); mStoreImp->mSpells.eraseStatic(spell.mId);
mSpells.insertStatic(spell); mStoreImp->mSpells.insertStatic(spell);
} }
} }
void ESMStore::movePlayerRecord()
{
auto player = mStoreImp->mNpcs.find("player");
mStoreImp->mNpcs.insert(*player);
}
void ESMStore::validateDynamic() void ESMStore::validateDynamic()
{ {
std::vector<ESM::NPC> npcsToReplace = getNPCsToReplace(mFactions, mClasses, mNpcs.mDynamic); std::vector<ESM::NPC> npcsToReplace = getNPCsToReplace(mStoreImp->mFactions, mStoreImp->mClasses, mStoreImp->mNpcs.mDynamic);
for (const ESM::NPC &npc : npcsToReplace) for (const ESM::NPC &npc : npcsToReplace)
mNpcs.insert(npc); mStoreImp->mNpcs.insert(npc);
removeMissingScripts(mScripts, mArmors.mDynamic); removeMissingScripts(mStoreImp->mScripts, mStoreImp->mArmors.mDynamic);
removeMissingScripts(mScripts, mBooks.mDynamic); removeMissingScripts(mStoreImp->mScripts, mStoreImp->mBooks.mDynamic);
removeMissingScripts(mScripts, mClothes.mDynamic); removeMissingScripts(mStoreImp->mScripts, mStoreImp->mClothes.mDynamic);
removeMissingScripts(mScripts, mWeapons.mDynamic); removeMissingScripts(mStoreImp->mScripts, mStoreImp->mWeapons.mDynamic);
removeMissingObjects(mCreatureLists); removeMissingObjects(mStoreImp->mCreatureLists);
removeMissingObjects(mItemLists); removeMissingObjects(mStoreImp->mItemLists);
} }
// Leveled lists can be modified by scripts. This removes items that no longer exist (presumably because the plugin was removed) from modified lists // Leveled lists can be modified by scripts. This removes items that no longer exist (presumably because the plugin was removed) from modified lists
@ -426,19 +535,19 @@ void ESMStore::removeMissingObjects(Store<T>& store)
int ESMStore::countSavedGameRecords() const int ESMStore::countSavedGameRecords() const
{ {
return 1 // DYNA (dynamic name counter) return 1 // DYNA (dynamic name counter)
+mPotions.getDynamicSize() +mStoreImp->mPotions.getDynamicSize()
+mArmors.getDynamicSize() +mStoreImp->mArmors.getDynamicSize()
+mBooks.getDynamicSize() +mStoreImp->mBooks.getDynamicSize()
+mClasses.getDynamicSize() +mStoreImp->mClasses.getDynamicSize()
+mClothes.getDynamicSize() +mStoreImp->mClothes.getDynamicSize()
+mEnchants.getDynamicSize() +mStoreImp->mEnchants.getDynamicSize()
+mNpcs.getDynamicSize() +mStoreImp->mNpcs.getDynamicSize()
+mSpells.getDynamicSize() +mStoreImp->mSpells.getDynamicSize()
+mWeapons.getDynamicSize() +mStoreImp->mWeapons.getDynamicSize()
+mCreatureLists.getDynamicSize() +mStoreImp->mCreatureLists.getDynamicSize()
+mItemLists.getDynamicSize() +mStoreImp->mItemLists.getDynamicSize()
+mCreatures.getDynamicSize() +mStoreImp->mCreatures.getDynamicSize()
+mContainers.getDynamicSize(); +mStoreImp->mContainers.getDynamicSize();
} }
void ESMStore::write (ESM::ESMWriter& writer, Loading::Listener& progress) const void ESMStore::write (ESM::ESMWriter& writer, Loading::Listener& progress) const
@ -449,19 +558,19 @@ void ESMStore::removeMissingObjects(Store<T>& store)
writer.endRecord("COUN"); writer.endRecord("COUN");
writer.endRecord(ESM::REC_DYNA); writer.endRecord(ESM::REC_DYNA);
mPotions.write (writer, progress); mStoreImp->mPotions.write (writer, progress);
mArmors.write (writer, progress); mStoreImp->mArmors.write (writer, progress);
mBooks.write (writer, progress); mStoreImp->mBooks.write (writer, progress);
mClasses.write (writer, progress); mStoreImp->mClasses.write (writer, progress);
mClothes.write (writer, progress); mStoreImp->mClothes.write (writer, progress);
mEnchants.write (writer, progress); mStoreImp->mEnchants.write (writer, progress);
mSpells.write (writer, progress); mStoreImp->mSpells.write (writer, progress);
mWeapons.write (writer, progress); mStoreImp->mWeapons.write (writer, progress);
mNpcs.write (writer, progress); mStoreImp->mNpcs.write (writer, progress);
mItemLists.write (writer, progress); mStoreImp->mItemLists.write (writer, progress);
mCreatureLists.write (writer, progress); mStoreImp->mCreatureLists.write (writer, progress);
mCreatures.write (writer, progress); mStoreImp->mCreatures.write (writer, progress);
mContainers.write (writer, progress); mStoreImp->mContainers.write (writer, progress);
} }
bool ESMStore::readRecord (ESM::ESMReader& reader, uint32_t type) bool ESMStore::readRecord (ESM::ESMReader& reader, uint32_t type)
@ -501,10 +610,10 @@ void ESMStore::removeMissingObjects(Store<T>& store)
{ {
setUp(); setUp();
const ESM::NPC *player = mNpcs.find ("player"); const ESM::NPC *player = mStoreImp->mNpcs.find ("player");
if (!mRaces.find (player->mRace) || if (!mStoreImp->mRaces.find (player->mRace) ||
!mClasses.find (player->mClass)) !mStoreImp->mClasses.find (player->mClass))
throw std::runtime_error ("Invalid player record (race or class unavailable"); throw std::runtime_error ("Invalid player record (race or class unavailable");
} }
@ -526,4 +635,242 @@ void ESMStore::removeMissingObjects(Store<T>& store)
} }
return {ptr, true}; return {ptr, true};
} }
template <>
const ESM::Cell *ESMStore::insert<ESM::Cell>(const ESM::Cell &cell) {
return mStoreImp->mCells.insert(cell);
}
template <>
const ESM::NPC *ESMStore::insert<ESM::NPC>(const ESM::NPC &npc)
{
const std::string id = "$dynamic" + std::to_string(mDynamicCount++);
if (Misc::StringUtils::ciEqual(npc.mId, "player"))
{
return mStoreImp->mNpcs.insert(npc);
}
else if (mStoreImp->mNpcs.search(id) != nullptr)
{
const std::string msg = "Try to override existing record '" + id + "'";
throw std::runtime_error(msg);
}
ESM::NPC record = npc;
record.mId = id;
ESM::NPC *ptr = mStoreImp->mNpcs.insert(record);
mIds[ptr->mId] = ESM::REC_NPC_;
return ptr;
}
template <>
const Store<ESM::Activator> &ESMStore::get<ESM::Activator>() const {
return mStoreImp->mActivators;
}
template <>
const Store<ESM::Potion> &ESMStore::get<ESM::Potion>() const {
return mStoreImp->mPotions;
}
template <>
const Store<ESM::Apparatus> &ESMStore::get<ESM::Apparatus>() const {
return mStoreImp->mAppas;
}
template <>
const Store<ESM::Armor> &ESMStore::get<ESM::Armor>() const {
return mStoreImp->mArmors;
}
template <>
const Store<ESM::BodyPart> &ESMStore::get<ESM::BodyPart>() const {
return mStoreImp->mBodyParts;
}
template <>
const Store<ESM::Book> &ESMStore::get<ESM::Book>() const {
return mStoreImp->mBooks;
}
template <>
const Store<ESM::BirthSign> &ESMStore::get<ESM::BirthSign>() const {
return mStoreImp->mBirthSigns;
}
template <>
const Store<ESM::Class> &ESMStore::get<ESM::Class>() const {
return mStoreImp->mClasses;
}
template <>
const Store<ESM::Clothing> &ESMStore::get<ESM::Clothing>() const {
return mStoreImp->mClothes;
}
template <>
const Store<ESM::Container> &ESMStore::get<ESM::Container>() const {
return mStoreImp->mContainers;
}
template <>
const Store<ESM::Creature> &ESMStore::get<ESM::Creature>() const {
return mStoreImp->mCreatures;
}
template <>
const Store<ESM::Dialogue> &ESMStore::get<ESM::Dialogue>() const {
return mStoreImp->mDialogs;
}
template <>
const Store<ESM::Door> &ESMStore::get<ESM::Door>() const {
return mStoreImp->mDoors;
}
template <>
const Store<ESM::Enchantment> &ESMStore::get<ESM::Enchantment>() const {
return mStoreImp->mEnchants;
}
template <>
const Store<ESM::Faction> &ESMStore::get<ESM::Faction>() const {
return mStoreImp->mFactions;
}
template <>
const Store<ESM::Global> &ESMStore::get<ESM::Global>() const {
return mStoreImp->mGlobals;
}
template <>
const Store<ESM::Ingredient> &ESMStore::get<ESM::Ingredient>() const {
return mStoreImp->mIngreds;
}
template <>
const Store<ESM::CreatureLevList> &ESMStore::get<ESM::CreatureLevList>() const {
return mStoreImp->mCreatureLists;
}
template <>
const Store<ESM::ItemLevList> &ESMStore::get<ESM::ItemLevList>() const {
return mStoreImp->mItemLists;
}
template <>
const Store<ESM::Light> &ESMStore::get<ESM::Light>() const {
return mStoreImp->mLights;
}
template <>
const Store<ESM::Lockpick> &ESMStore::get<ESM::Lockpick>() const {
return mStoreImp->mLockpicks;
}
template <>
const Store<ESM::Miscellaneous> &ESMStore::get<ESM::Miscellaneous>() const {
return mStoreImp->mMiscItems;
}
template <>
const Store<ESM::NPC> &ESMStore::get<ESM::NPC>() const {
return mStoreImp->mNpcs;
}
template <>
const Store<ESM::Probe> &ESMStore::get<ESM::Probe>() const {
return mStoreImp->mProbes;
}
template <>
const Store<ESM::Race> &ESMStore::get<ESM::Race>() const {
return mStoreImp->mRaces;
}
template <>
const Store<ESM::Region> &ESMStore::get<ESM::Region>() const {
return mStoreImp->mRegions;
}
template <>
const Store<ESM::Repair> &ESMStore::get<ESM::Repair>() const {
return mStoreImp->mRepairs;
}
template <>
const Store<ESM::SoundGenerator> &ESMStore::get<ESM::SoundGenerator>() const {
return mStoreImp->mSoundGens;
}
template <>
const Store<ESM::Sound> &ESMStore::get<ESM::Sound>() const {
return mStoreImp->mSounds;
}
template <>
const Store<ESM::Spell> &ESMStore::get<ESM::Spell>() const {
return mStoreImp->mSpells;
}
template <>
const Store<ESM::StartScript> &ESMStore::get<ESM::StartScript>() const {
return mStoreImp->mStartScripts;
}
template <>
const Store<ESM::Static> &ESMStore::get<ESM::Static>() const {
return mStoreImp->mStatics;
}
template <>
const Store<ESM::Weapon> &ESMStore::get<ESM::Weapon>() const {
return mStoreImp->mWeapons;
}
template <>
const Store<ESM::GameSetting> &ESMStore::get<ESM::GameSetting>() const {
return mStoreImp->mGameSettings;
}
template <>
const Store<ESM::Script> &ESMStore::get<ESM::Script>() const {
return mStoreImp->mScripts;
}
template <>
const Store<ESM::Cell> &ESMStore::get<ESM::Cell>() const {
return mStoreImp->mCells;
}
template <>
const Store<ESM::Land> &ESMStore::get<ESM::Land>() const {
return mStoreImp->mLands;
}
template <>
const Store<ESM::LandTexture> &ESMStore::get<ESM::LandTexture>() const {
return mStoreImp->mLandTextures;
}
template <>
const Store<ESM::Pathgrid> &ESMStore::get<ESM::Pathgrid>() const {
return mStoreImp->mPathgrids;
}
template <>
const Store<ESM::MagicEffect> &ESMStore::get<ESM::MagicEffect>() const {
return mStoreImp->mMagicEffects;
}
template <>
const Store<ESM::Skill> &ESMStore::get<ESM::Skill>() const {
return mStoreImp->mSkills;
}
template <>
const Store<ESM::Attribute> &ESMStore::get<ESM::Attribute>() const {
return mStoreImp->mAttributes;
}
} // end namespace } // end namespace

View file

@ -6,7 +6,6 @@
#include <unordered_map> #include <unordered_map>
#include <components/esm/luascripts.hpp> #include <components/esm/luascripts.hpp>
#include <components/esm/records.hpp>
#include "store.hpp" #include "store.hpp"
namespace Loading namespace Loading
@ -26,56 +25,12 @@ namespace ESM
namespace MWWorld namespace MWWorld
{ {
struct ESMStoreImp;
class ESMStore class ESMStore
{ {
Store<ESM::Activator> mActivators; std::unique_ptr<ESMStoreImp> mStoreImp;
Store<ESM::Potion> mPotions;
Store<ESM::Apparatus> mAppas;
Store<ESM::Armor> mArmors;
Store<ESM::BodyPart> mBodyParts;
Store<ESM::Book> mBooks;
Store<ESM::BirthSign> mBirthSigns;
Store<ESM::Class> mClasses;
Store<ESM::Clothing> mClothes;
Store<ESM::Container> mContainers;
Store<ESM::Creature> mCreatures;
Store<ESM::Dialogue> mDialogs;
Store<ESM::Door> mDoors;
Store<ESM::Enchantment> mEnchants;
Store<ESM::Faction> mFactions;
Store<ESM::Global> mGlobals;
Store<ESM::Ingredient> mIngreds;
Store<ESM::CreatureLevList> mCreatureLists;
Store<ESM::ItemLevList> mItemLists;
Store<ESM::Light> mLights;
Store<ESM::Lockpick> mLockpicks;
Store<ESM::Miscellaneous> mMiscItems;
Store<ESM::NPC> mNpcs;
Store<ESM::Probe> mProbes;
Store<ESM::Race> mRaces;
Store<ESM::Region> mRegions;
Store<ESM::Repair> mRepairs;
Store<ESM::SoundGenerator> mSoundGens;
Store<ESM::Sound> mSounds;
Store<ESM::Spell> mSpells;
Store<ESM::StartScript> mStartScripts;
Store<ESM::Static> mStatics;
Store<ESM::Weapon> mWeapons;
Store<ESM::GameSetting> mGameSettings;
Store<ESM::Script> mScripts;
// Lists that need special rules
Store<ESM::Cell> mCells;
Store<ESM::Land> mLands;
Store<ESM::LandTexture> mLandTextures;
Store<ESM::Pathgrid> mPathgrids;
Store<ESM::MagicEffect> mMagicEffects;
Store<ESM::Skill> mSkills;
// Special entry which is hardcoded and not loaded from an ESM
Store<ESM::Attribute> mAttributes;
// Lookup of all IDs. Makes looking up references faster. Just // Lookup of all IDs. Makes looking up references faster. Just
// maps the id name to the record type. // maps the id name to the record type.
@ -137,51 +92,8 @@ namespace MWWorld
return it->second; return it->second;
} }
ESMStore() ESMStore();
: mDynamicCount(0) ~ESMStore();
{
mStores[ESM::REC_ACTI] = &mActivators;
mStores[ESM::REC_ALCH] = &mPotions;
mStores[ESM::REC_APPA] = &mAppas;
mStores[ESM::REC_ARMO] = &mArmors;
mStores[ESM::REC_BODY] = &mBodyParts;
mStores[ESM::REC_BOOK] = &mBooks;
mStores[ESM::REC_BSGN] = &mBirthSigns;
mStores[ESM::REC_CELL] = &mCells;
mStores[ESM::REC_CLAS] = &mClasses;
mStores[ESM::REC_CLOT] = &mClothes;
mStores[ESM::REC_CONT] = &mContainers;
mStores[ESM::REC_CREA] = &mCreatures;
mStores[ESM::REC_DIAL] = &mDialogs;
mStores[ESM::REC_DOOR] = &mDoors;
mStores[ESM::REC_ENCH] = &mEnchants;
mStores[ESM::REC_FACT] = &mFactions;
mStores[ESM::REC_GLOB] = &mGlobals;
mStores[ESM::REC_GMST] = &mGameSettings;
mStores[ESM::REC_INGR] = &mIngreds;
mStores[ESM::REC_LAND] = &mLands;
mStores[ESM::REC_LEVC] = &mCreatureLists;
mStores[ESM::REC_LEVI] = &mItemLists;
mStores[ESM::REC_LIGH] = &mLights;
mStores[ESM::REC_LOCK] = &mLockpicks;
mStores[ESM::REC_LTEX] = &mLandTextures;
mStores[ESM::REC_MISC] = &mMiscItems;
mStores[ESM::REC_NPC_] = &mNpcs;
mStores[ESM::REC_PGRD] = &mPathgrids;
mStores[ESM::REC_PROB] = &mProbes;
mStores[ESM::REC_RACE] = &mRaces;
mStores[ESM::REC_REGN] = &mRegions;
mStores[ESM::REC_REPA] = &mRepairs;
mStores[ESM::REC_SCPT] = &mScripts;
mStores[ESM::REC_SNDG] = &mSoundGens;
mStores[ESM::REC_SOUN] = &mSounds;
mStores[ESM::REC_SPEL] = &mSpells;
mStores[ESM::REC_SSCR] = &mStartScripts;
mStores[ESM::REC_STAT] = &mStatics;
mStores[ESM::REC_WEAP] = &mWeapons;
mPathgrids.setCells(mCells);
}
void clearDynamic () void clearDynamic ()
{ {
@ -191,11 +103,7 @@ namespace MWWorld
movePlayerRecord(); movePlayerRecord();
} }
void movePlayerRecord () void movePlayerRecord();
{
auto player = mNpcs.find("player");
mNpcs.insert(*player);
}
/// Validate entries in store after loading a save /// Validate entries in store after loading a save
void validateDynamic(); void validateDynamic();
@ -289,242 +197,136 @@ namespace MWWorld
}; };
template <> template <>
inline const ESM::Cell *ESMStore::insert<ESM::Cell>(const ESM::Cell &cell) { const ESM::Cell* ESMStore::insert<ESM::Cell>(const ESM::Cell& cell);
return mCells.insert(cell);
}
template <> template <>
inline const ESM::NPC *ESMStore::insert<ESM::NPC>(const ESM::NPC &npc) const ESM::NPC* ESMStore::insert<ESM::NPC>(const ESM::NPC& npc);
{
if (Misc::StringUtils::ciEqual(npc.mId, "player"))
{
return mNpcs.insert(npc);
}
const std::string id = "$dynamic" + std::to_string(mDynamicCount++);
if (mNpcs.search(id) != nullptr)
{
const std::string msg = "Try to override existing record '" + id + "'";
throw std::runtime_error(msg);
}
ESM::NPC record = npc;
record.mId = id;
ESM::NPC *ptr = mNpcs.insert(record);
mIds[ptr->mId] = ESM::REC_NPC_;
return ptr;
}
template <> template <>
inline const Store<ESM::Activator> &ESMStore::get<ESM::Activator>() const { const Store<ESM::Activator>& ESMStore::get<ESM::Activator>() const;
return mActivators;
}
template <> template <>
inline const Store<ESM::Potion> &ESMStore::get<ESM::Potion>() const { const Store<ESM::Potion>& ESMStore::get<ESM::Potion>() const;
return mPotions;
}
template <> template <>
inline const Store<ESM::Apparatus> &ESMStore::get<ESM::Apparatus>() const { const Store<ESM::Apparatus>& ESMStore::get<ESM::Apparatus>() const;
return mAppas;
}
template <> template <>
inline const Store<ESM::Armor> &ESMStore::get<ESM::Armor>() const { const Store<ESM::Armor>& ESMStore::get<ESM::Armor>() const;
return mArmors;
}
template <> template <>
inline const Store<ESM::BodyPart> &ESMStore::get<ESM::BodyPart>() const { const Store<ESM::BodyPart>& ESMStore::get<ESM::BodyPart>() const;
return mBodyParts;
}
template <> template <>
inline const Store<ESM::Book> &ESMStore::get<ESM::Book>() const { const Store<ESM::Book>& ESMStore::get<ESM::Book>() const;
return mBooks;
}
template <> template <>
inline const Store<ESM::BirthSign> &ESMStore::get<ESM::BirthSign>() const { const Store<ESM::BirthSign>& ESMStore::get<ESM::BirthSign>() const;
return mBirthSigns;
}
template <> template <>
inline const Store<ESM::Class> &ESMStore::get<ESM::Class>() const { const Store<ESM::Class>& ESMStore::get<ESM::Class>() const;
return mClasses;
}
template <> template <>
inline const Store<ESM::Clothing> &ESMStore::get<ESM::Clothing>() const { const Store<ESM::Clothing>& ESMStore::get<ESM::Clothing>() const;
return mClothes;
}
template <> template <>
inline const Store<ESM::Container> &ESMStore::get<ESM::Container>() const { const Store<ESM::Container>& ESMStore::get<ESM::Container>() const;
return mContainers;
}
template <> template <>
inline const Store<ESM::Creature> &ESMStore::get<ESM::Creature>() const { const Store<ESM::Creature>& ESMStore::get<ESM::Creature>() const;
return mCreatures;
}
template <> template <>
inline const Store<ESM::Dialogue> &ESMStore::get<ESM::Dialogue>() const { const Store<ESM::Dialogue>& ESMStore::get<ESM::Dialogue>() const;
return mDialogs;
}
template <> template <>
inline const Store<ESM::Door> &ESMStore::get<ESM::Door>() const { const Store<ESM::Door>& ESMStore::get<ESM::Door>() const;
return mDoors;
}
template <> template <>
inline const Store<ESM::Enchantment> &ESMStore::get<ESM::Enchantment>() const { const Store<ESM::Enchantment>& ESMStore::get<ESM::Enchantment>() const;
return mEnchants;
}
template <> template <>
inline const Store<ESM::Faction> &ESMStore::get<ESM::Faction>() const { const Store<ESM::Faction>& ESMStore::get<ESM::Faction>() const;
return mFactions;
}
template <> template <>
inline const Store<ESM::Global> &ESMStore::get<ESM::Global>() const { const Store<ESM::Global>& ESMStore::get<ESM::Global>() const;
return mGlobals;
}
template <> template <>
inline const Store<ESM::Ingredient> &ESMStore::get<ESM::Ingredient>() const { const Store<ESM::Ingredient>& ESMStore::get<ESM::Ingredient>() const;
return mIngreds;
}
template <> template <>
inline const Store<ESM::CreatureLevList> &ESMStore::get<ESM::CreatureLevList>() const { const Store<ESM::CreatureLevList>& ESMStore::get<ESM::CreatureLevList>() const;
return mCreatureLists;
}
template <> template <>
inline const Store<ESM::ItemLevList> &ESMStore::get<ESM::ItemLevList>() const { const Store<ESM::ItemLevList>& ESMStore::get<ESM::ItemLevList>() const;
return mItemLists;
}
template <> template <>
inline const Store<ESM::Light> &ESMStore::get<ESM::Light>() const { const Store<ESM::Light>& ESMStore::get<ESM::Light>() const;
return mLights;
}
template <> template <>
inline const Store<ESM::Lockpick> &ESMStore::get<ESM::Lockpick>() const { const Store<ESM::Lockpick>& ESMStore::get<ESM::Lockpick>() const;
return mLockpicks;
}
template <> template <>
inline const Store<ESM::Miscellaneous> &ESMStore::get<ESM::Miscellaneous>() const { const Store<ESM::Miscellaneous>& ESMStore::get<ESM::Miscellaneous>() const;
return mMiscItems;
}
template <> template <>
inline const Store<ESM::NPC> &ESMStore::get<ESM::NPC>() const { const Store<ESM::NPC>& ESMStore::get<ESM::NPC>() const;
return mNpcs;
}
template <> template <>
inline const Store<ESM::Probe> &ESMStore::get<ESM::Probe>() const { const Store<ESM::Probe>& ESMStore::get<ESM::Probe>() const;
return mProbes;
}
template <> template <>
inline const Store<ESM::Race> &ESMStore::get<ESM::Race>() const { const Store<ESM::Race>& ESMStore::get<ESM::Race>() const;
return mRaces;
}
template <> template <>
inline const Store<ESM::Region> &ESMStore::get<ESM::Region>() const { const Store<ESM::Region>& ESMStore::get<ESM::Region>() const;
return mRegions;
}
template <> template <>
inline const Store<ESM::Repair> &ESMStore::get<ESM::Repair>() const { const Store<ESM::Repair>& ESMStore::get<ESM::Repair>() const;
return mRepairs;
}
template <> template <>
inline const Store<ESM::SoundGenerator> &ESMStore::get<ESM::SoundGenerator>() const { const Store<ESM::SoundGenerator>& ESMStore::get<ESM::SoundGenerator>() const;
return mSoundGens;
}
template <> template <>
inline const Store<ESM::Sound> &ESMStore::get<ESM::Sound>() const { const Store<ESM::Sound>& ESMStore::get<ESM::Sound>() const;
return mSounds;
}
template <> template <>
inline const Store<ESM::Spell> &ESMStore::get<ESM::Spell>() const { const Store<ESM::Spell>& ESMStore::get<ESM::Spell>() const;
return mSpells;
}
template <> template <>
inline const Store<ESM::StartScript> &ESMStore::get<ESM::StartScript>() const { const Store<ESM::StartScript>& ESMStore::get<ESM::StartScript>() const;
return mStartScripts;
}
template <> template <>
inline const Store<ESM::Static> &ESMStore::get<ESM::Static>() const { const Store<ESM::Static>& ESMStore::get<ESM::Static>() const;
return mStatics;
}
template <> template <>
inline const Store<ESM::Weapon> &ESMStore::get<ESM::Weapon>() const { const Store<ESM::Weapon>& ESMStore::get<ESM::Weapon>() const;
return mWeapons;
}
template <> template <>
inline const Store<ESM::GameSetting> &ESMStore::get<ESM::GameSetting>() const { const Store<ESM::GameSetting>& ESMStore::get<ESM::GameSetting>() const;
return mGameSettings;
}
template <> template <>
inline const Store<ESM::Script> &ESMStore::get<ESM::Script>() const { const Store<ESM::Script>& ESMStore::get<ESM::Script>() const;
return mScripts;
}
template <> template <>
inline const Store<ESM::Cell> &ESMStore::get<ESM::Cell>() const { const Store<ESM::Cell>& ESMStore::get<ESM::Cell>() const;
return mCells;
}
template <> template <>
inline const Store<ESM::Land> &ESMStore::get<ESM::Land>() const { const Store<ESM::Land>& ESMStore::get<ESM::Land>() const;
return mLands;
}
template <> template <>
inline const Store<ESM::LandTexture> &ESMStore::get<ESM::LandTexture>() const { const Store<ESM::LandTexture>& ESMStore::get<ESM::LandTexture>() const;
return mLandTextures;
}
template <> template <>
inline const Store<ESM::Pathgrid> &ESMStore::get<ESM::Pathgrid>() const { const Store<ESM::Pathgrid>& ESMStore::get<ESM::Pathgrid>() const;
return mPathgrids;
}
template <> template <>
inline const Store<ESM::MagicEffect> &ESMStore::get<ESM::MagicEffect>() const { const Store<ESM::MagicEffect>& ESMStore::get<ESM::MagicEffect>() const;
return mMagicEffects;
}
template <> template <>
inline const Store<ESM::Skill> &ESMStore::get<ESM::Skill>() const { const Store<ESM::Skill>& ESMStore::get<ESM::Skill>() const;
return mSkills;
}
template <> template <>
inline const Store<ESM::Attribute> &ESMStore::get<ESM::Attribute>() const { const Store<ESM::Attribute>& ESMStore::get<ESM::Attribute>() const;
return mAttributes;
}
} }
#endif #endif