openmw-tes3coop/apps/openmw/mwworld/esmstore.cpp

150 lines
5.1 KiB
C++
Raw Normal View History

2012-10-01 15:17:04 +00:00
#include "esmstore.hpp"
2010-05-17 18:59:15 +00:00
#include <set>
#include <iostream>
#include <boost/filesystem/v3/operations.hpp>
2012-10-01 15:17:04 +00:00
namespace MWWorld
{
2012-11-05 14:09:14 +00:00
static bool isCacheableRecord(int id)
{
if (id == ESM::REC_ACTI || id == ESM::REC_ALCH || id == ESM::REC_APPA || id == ESM::REC_ARMO ||
id == ESM::REC_BOOK || id == ESM::REC_CLOT || id == ESM::REC_CONT || id == ESM::REC_CREA ||
id == ESM::REC_DOOR || id == ESM::REC_INGR || id == ESM::REC_LEVC || id == ESM::REC_LEVI ||
id == ESM::REC_LIGH || id == ESM::REC_LOCK || id == ESM::REC_MISC || id == ESM::REC_NPC_ ||
id == ESM::REC_PROB || id == ESM::REC_REPA || id == ESM::REC_STAT || id == ESM::REC_WEAP)
{
return true;
}
return false;
}
2012-10-01 15:17:04 +00:00
void ESMStore::load(ESM::ESMReader &esm)
{
2012-10-01 15:17:04 +00:00
std::set<std::string> missing;
2010-08-06 13:19:39 +00:00
ESM::Dialogue *dialogue = 0;
// Cache parent esX files by tracking their indices in the global list of
// all files/readers used by the engine. This will greaty help to accelerate
// parsing of reference IDs.
size_t index = ~0;
const ESM::ESMReader::MasterList &masters = esm.getMasters();
std::vector<ESM::ESMReader> *allPlugins = esm.getGlobalReaderList();
for (size_t j = 0; j < masters.size(); j++) {
ESM::MasterData &mast = const_cast<ESM::MasterData&>(masters[j]);
std::string fname = mast.name;
for (size_t i = 0; i < esm.getIndex(); i++) {
const std::string &candidate = allPlugins->at(i).getContext().filename;
std::string fnamecandidate = boost::filesystem::path(candidate).filename().string();
if (fname == fnamecandidate) {
index = i;
break;
}
}
if (index == (size_t)~0) {
// Tried to load a parent file that has not been loaded yet. This is bad,
// the launcher should have taken care of this.
std::string fstring = "File " + fname + " asks for parent file " + masters[j].name
+ ", but it has not been loaded yet. Please check your load order.";
esm.fail(fstring);
}
mast.index = index;
}
2010-08-06 13:19:39 +00:00
// Loop through all records
while(esm.hasMoreRecs())
{
2012-10-01 15:17:04 +00:00
ESM::NAME n = esm.getRecName();
2010-08-06 13:19:39 +00:00
esm.getRecHeader();
2010-08-06 13:19:39 +00:00
// Look up the record type.
2012-11-05 14:09:14 +00:00
std::map<int, StoreBase *>::iterator it = mStores.find(n.val);
if (it == mStores.end()) {
if (n.val == ESM::REC_INFO) {
if (dialogue) {
dialogue->mInfo.push_back(ESM::DialInfo());
dialogue->mInfo.back().load(esm);
} else {
2010-08-06 13:19:39 +00:00
std::cerr << "error: info record without dialog" << std::endl;
esm.skipRecord();
}
2012-11-05 14:09:14 +00:00
} else if (n.val == ESM::REC_MGEF) {
mMagicEffects.load (esm);
} else if (n.val == ESM::REC_SKIL) {
mSkills.load (esm);
} else {
2010-08-06 13:19:39 +00:00
// Not found (this would be an error later)
esm.skipRecord();
missing.insert(n.toString());
}
2012-11-05 14:09:14 +00:00
} else {
2010-08-06 13:19:39 +00:00
// Load it
std::string id = esm.getHNOString("NAME");
// ... unless it got deleted! This means that the following record
// has been deleted, and trying to load it using standard assumptions
// on the structure will (probably) fail.
if (esm.isNextSub("DELE")) {
esm.skipRecord();
all.erase(id);
it->second->remove(id);
continue;
}
2010-08-06 13:19:39 +00:00
it->second->load(esm, id);
2012-11-05 14:09:14 +00:00
if (n.val==ESM::REC_DIAL) {
// dirty hack, but it is better than non-const search()
// or friends
2012-11-06 11:26:55 +00:00
dialogue = &mDialogs.mStatic.back();
assert (dialogue->mId == id);
2012-11-05 14:09:14 +00:00
} else {
2010-08-06 13:19:39 +00:00
dialogue = 0;
2012-11-05 14:09:14 +00:00
}
2010-08-06 13:19:39 +00:00
// Insert the reference into the global lookup
2012-11-05 14:09:14 +00:00
if (!id.empty() && isCacheableRecord(n.val)) {
mIds[id] = n.val;
}
2010-08-06 13:19:39 +00:00
}
}
/* This information isn't needed on screen. But keep the code around
for debugging purposes later.
2012-11-05 14:09:14 +00:00
cout << "\n" << mStores.size() << " record types:\n";
for(RecListList::iterator it = mStores.begin(); it != mStores.end(); it++)
2010-05-17 18:59:15 +00:00
cout << " " << toStr(it->first) << ": " << it->second->getSize() << endl;
cout << "\nNot implemented yet: ";
for(set<string>::iterator it = missing.begin();
it != missing.end(); it++ )
cout << *it << " ";
cout << endl;
*/
//setUp();
}
2012-10-01 15:17:04 +00:00
2012-11-05 14:09:14 +00:00
void ESMStore::setUp()
{
std::map<int, StoreBase *>::iterator it = mStores.begin();
for (; it != mStores.end(); ++it) {
it->second->setUp();
}
mSkills.setUp();
mMagicEffects.setUp();
mAttributes.setUp();
2012-11-06 14:17:32 +00:00
ESM::NPC item;
item.mId = "player";
std::vector<ESM::NPC>::iterator pIt =
std::lower_bound(mNpcs.mStatic.begin(), mNpcs.mStatic.end(), item, RecordCmp());
assert(pIt != mNpcs.mStatic.end() && pIt->mId == "player");
mNpcs.insert(*pIt);
mNpcs.mStatic.erase(pIt);
2012-11-05 14:09:14 +00:00
}
2012-10-01 15:17:04 +00:00
} // end namespace