2012-10-01 19:17:04 +04:00
|
|
|
#include "esmstore.hpp"
|
|
|
|
|
2010-05-17 20:59:15 +02:00
|
|
|
#include <set>
|
2010-05-17 17:35:42 +02:00
|
|
|
#include <iostream>
|
|
|
|
|
2013-02-14 04:22:00 -05:00
|
|
|
#include <boost/filesystem/operations.hpp>
|
2012-11-25 17:19:29 +01:00
|
|
|
|
2013-08-27 15:48:13 +02:00
|
|
|
#include <components/loadinglistener/loadinglistener.hpp>
|
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
namespace MWWorld
|
2010-05-17 17:35:42 +02:00
|
|
|
{
|
|
|
|
|
2012-11-05 18:09:14 +04: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;
|
|
|
|
}
|
|
|
|
|
2013-08-27 15:48:13 +02:00
|
|
|
void ESMStore::load(ESM::ESMReader &esm, Loading::Listener* listener)
|
2010-05-17 17:35:42 +02:00
|
|
|
{
|
2013-08-27 15:48:13 +02:00
|
|
|
listener->setProgressRange(1000);
|
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
std::set<std::string> missing;
|
2010-05-17 17:35:42 +02:00
|
|
|
|
2010-08-06 15:19:39 +02:00
|
|
|
ESM::Dialogue *dialogue = 0;
|
|
|
|
|
2013-03-12 08:15:20 +01:00
|
|
|
/// \todo Move this to somewhere else. ESMReader?
|
2012-11-25 17:19:29 +01:00
|
|
|
// Cache parent esX files by tracking their indices in the global list of
|
2013-01-20 19:07:33 +01:00
|
|
|
// all files/readers used by the engine. This will greaty accelerate
|
|
|
|
// refnumber mangling, as required for handling moved references.
|
2013-01-19 23:33:18 +01:00
|
|
|
int index = ~0;
|
2013-09-21 23:06:29 -05:00
|
|
|
const std::vector<ESM::Header::MasterData> &masters = esm.getGameFiles();
|
2012-11-25 17:19:29 +01:00
|
|
|
std::vector<ESM::ESMReader> *allPlugins = esm.getGlobalReaderList();
|
|
|
|
for (size_t j = 0; j < masters.size(); j++) {
|
2013-03-12 08:15:20 +01:00
|
|
|
ESM::Header::MasterData &mast = const_cast<ESM::Header::MasterData&>(masters[j]);
|
2012-11-25 17:19:29 +01:00
|
|
|
std::string fname = mast.name;
|
2013-01-19 23:33:18 +01:00
|
|
|
for (int i = 0; i < esm.getIndex(); i++) {
|
2012-11-25 17:19:29 +01:00
|
|
|
const std::string &candidate = allPlugins->at(i).getContext().filename;
|
|
|
|
std::string fnamecandidate = boost::filesystem::path(candidate).filename().string();
|
|
|
|
if (fname == fnamecandidate) {
|
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-01-19 23:33:18 +01:00
|
|
|
if (index == (int)~0) {
|
2012-11-25 17:19:29 +01:00
|
|
|
// Tried to load a parent file that has not been loaded yet. This is bad,
|
|
|
|
// the launcher should have taken care of this.
|
2013-12-16 11:39:24 +01:00
|
|
|
std::string fstring = "File " + esm.getName() + " asks for parent file " + masters[j].name
|
2012-11-25 17:19:29 +01:00
|
|
|
+ ", but it has not been loaded yet. Please check your load order.";
|
|
|
|
esm.fail(fstring);
|
|
|
|
}
|
|
|
|
mast.index = index;
|
|
|
|
}
|
|
|
|
|
2010-08-06 15:19:39 +02:00
|
|
|
// Loop through all records
|
|
|
|
while(esm.hasMoreRecs())
|
2010-05-17 17:35:42 +02:00
|
|
|
{
|
2012-10-01 19:17:04 +04:00
|
|
|
ESM::NAME n = esm.getRecName();
|
2010-08-06 15:19:39 +02:00
|
|
|
esm.getRecHeader();
|
2010-05-17 17:35:42 +02:00
|
|
|
|
2010-08-06 15:19:39 +02:00
|
|
|
// Look up the record type.
|
2012-11-05 18:09:14 +04:00
|
|
|
std::map<int, StoreBase *>::iterator it = mStores.find(n.val);
|
|
|
|
|
|
|
|
if (it == mStores.end()) {
|
|
|
|
if (n.val == ESM::REC_INFO) {
|
2013-11-05 12:56:20 +01:00
|
|
|
std::string id = esm.getHNOString("INAM");
|
2012-11-05 18:09:14 +04:00
|
|
|
if (dialogue) {
|
|
|
|
dialogue->mInfo.push_back(ESM::DialInfo());
|
2013-11-05 12:56:20 +01:00
|
|
|
dialogue->mInfo.back().mId = id;
|
2012-11-05 18:09:14 +04:00
|
|
|
dialogue->mInfo.back().load(esm);
|
|
|
|
} else {
|
2010-08-06 15:19:39 +02:00
|
|
|
std::cerr << "error: info record without dialog" << std::endl;
|
|
|
|
esm.skipRecord();
|
|
|
|
}
|
2012-11-05 18:09:14 +04: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 15:19:39 +02:00
|
|
|
// Not found (this would be an error later)
|
|
|
|
esm.skipRecord();
|
|
|
|
missing.insert(n.toString());
|
|
|
|
}
|
2012-11-05 18:09:14 +04:00
|
|
|
} else {
|
2010-08-06 15:19:39 +02:00
|
|
|
// Load it
|
|
|
|
std::string id = esm.getHNOString("NAME");
|
2012-11-25 19:07:16 +01:00
|
|
|
// ... 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();
|
2012-12-26 10:34:59 +01:00
|
|
|
it->second->eraseStatic(id);
|
2012-11-25 19:07:16 +01:00
|
|
|
continue;
|
|
|
|
}
|
2010-08-06 15:19:39 +02:00
|
|
|
it->second->load(esm, id);
|
|
|
|
|
2012-11-05 18:09:14 +04:00
|
|
|
if (n.val==ESM::REC_DIAL) {
|
2012-12-26 10:34:59 +01:00
|
|
|
dialogue = const_cast<ESM::Dialogue*>(mDialogs.find(id));
|
2012-11-05 18:09:14 +04:00
|
|
|
} else {
|
2010-08-06 15:19:39 +02:00
|
|
|
dialogue = 0;
|
2012-11-05 18:09:14 +04:00
|
|
|
}
|
2010-08-06 15:19:39 +02:00
|
|
|
// Insert the reference into the global lookup
|
2012-11-05 18:09:14 +04:00
|
|
|
if (!id.empty() && isCacheableRecord(n.val)) {
|
2014-01-06 13:53:20 +01:00
|
|
|
mIds[Misc::StringUtils::lowerCase (id)] = n.val;
|
2012-11-05 18:09:14 +04:00
|
|
|
}
|
2010-08-06 15:19:39 +02:00
|
|
|
}
|
2013-08-27 15:48:13 +02:00
|
|
|
listener->setProgress(esm.getFileOffset() / (float)esm.getFileSize() * 1000);
|
2010-05-17 17:35:42 +02:00
|
|
|
}
|
|
|
|
|
2010-07-17 14:01:47 +02:00
|
|
|
/* This information isn't needed on screen. But keep the code around
|
|
|
|
for debugging purposes later.
|
|
|
|
|
2012-11-05 18:09:14 +04:00
|
|
|
cout << "\n" << mStores.size() << " record types:\n";
|
|
|
|
for(RecListList::iterator it = mStores.begin(); it != mStores.end(); it++)
|
2010-05-17 20:59:15 +02: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;
|
2010-07-17 14:01:47 +02:00
|
|
|
*/
|
2010-05-17 17:35:42 +02:00
|
|
|
}
|
2012-10-01 19:17:04 +04:00
|
|
|
|
2012-11-05 18:09:14 +04: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();
|
|
|
|
}
|
|
|
|
|
2013-12-07 13:17:28 +01:00
|
|
|
int ESMStore::countSavedGameRecords() const
|
|
|
|
{
|
|
|
|
return
|
|
|
|
mPotions.getDynamicSize()
|
|
|
|
+mArmors.getDynamicSize()
|
|
|
|
+mBooks.getDynamicSize()
|
|
|
|
+mClasses.getDynamicSize()
|
|
|
|
+mClothes.getDynamicSize()
|
|
|
|
+mEnchants.getDynamicSize()
|
|
|
|
+mNpcs.getDynamicSize()
|
|
|
|
+mSpells.getDynamicSize()
|
|
|
|
+mWeapons.getDynamicSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ESMStore::write (ESM::ESMWriter& writer) const
|
|
|
|
{
|
|
|
|
mPotions.write (writer);
|
|
|
|
mArmors.write (writer);
|
|
|
|
mBooks.write (writer);
|
|
|
|
mClasses.write (writer);
|
|
|
|
mClothes.write (writer);
|
|
|
|
mEnchants.write (writer);
|
|
|
|
mNpcs.write (writer);
|
|
|
|
mSpells.write (writer);
|
|
|
|
mWeapons.write (writer);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ESMStore::readRecord (ESM::ESMReader& reader, int32_t type)
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case ESM::REC_ALCH:
|
|
|
|
case ESM::REC_ARMO:
|
|
|
|
case ESM::REC_BOOK:
|
|
|
|
case ESM::REC_CLAS:
|
|
|
|
case ESM::REC_CLOT:
|
|
|
|
case ESM::REC_ENCH:
|
|
|
|
case ESM::REC_NPC_:
|
|
|
|
case ESM::REC_SPEL:
|
|
|
|
case ESM::REC_WEAP:
|
|
|
|
|
|
|
|
mStores[type]->read (reader);
|
|
|
|
return true;
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-01 19:17:04 +04:00
|
|
|
} // end namespace
|