Only loads ESM4::reference when they are needed

and only those from the cell they are a part of.
The cell stores where it starts in the file for quick access later.
depth-refraction
florent.teppe 2 years ago
parent 31ae1cd339
commit 214cb8d8fe

@ -42,7 +42,10 @@
#include <components/esm4/loadligh.hpp> #include <components/esm4/loadligh.hpp>
#include <components/esm4/loadrefr.hpp> #include <components/esm4/loadrefr.hpp>
#include <components/esm4/loadstat.hpp> #include <components/esm4/loadstat.hpp>
#include <components/esm4/readerutils.hpp>
#include <components/files/openfile.hpp>
#include <components/misc/tuplehelpers.hpp> #include <components/misc/tuplehelpers.hpp>
#include <components/resource/resourcesystem.hpp>
#include "../mwbase/environment.hpp" #include "../mwbase/environment.hpp"
#include "../mwbase/luamanager.hpp" #include "../mwbase/luamanager.hpp"
@ -756,19 +759,46 @@ namespace MWWorld
} }
} }
void CellStore::listRefs(const ESM4::Cell& cell) template <typename ReferenceInvlocable>
{ static void visitCell4References(const ESM4::Cell& cell, ReferenceInvlocable&& invocable)
auto& refs = MWBase::Environment::get().getWorld()->getStore().get<ESM4::Reference>(); {
auto stream = Files::openBinaryInputFileStream(cell.mReaderContext.filename);
for (const auto& ref : refs) ESM4::Reader readerESM4(
std::move(stream), cell.mReaderContext.filename, MWBase::Environment::get().getResourceSystem()->getVFS());
readerESM4.setEncoder(nullptr);
readerESM4.restoreContext(cell.mReaderContext);
bool continueRead = true;
while (ESM::RefId::formIdRefId(readerESM4.getContext().currCell) == cell.mId && readerESM4.hasMoreRecs()
&& continueRead)
{ {
if (ref.mParent == cell.mId) continueRead = ESM4::ReaderUtils::readItem(
{ readerESM4,
mIds.push_back(ref.mBaseObj); [&](ESM4::Reader& reader) {
} auto recordType = static_cast<ESM4::RecordTypes>(reader.hdr().record.typeId);
ESM::RecNameInts esm4RecName = static_cast<ESM::RecNameInts>(ESM::esm4Recname(recordType));
if (esm4RecName == ESM::RecNameInts::REC_REFR4)
{
ESM4::Reference ref;
ref.load(reader);
invocable(ref);
return true;
}
else if (esm4RecName == ESM::RecNameInts::REC_CELL4)
{
ESM4::Cell cellToLoad;
cellToLoad.load(reader); // This is necessary to exit
}
return false;
},
[&](ESM4::Reader& reader) {});
} }
} }
void CellStore::listRefs(const ESM4::Cell& cell)
{
visitCell4References(cell, [&](ESM4::Reference& ref) { mIds.push_back(ref.mBaseObj); });
}
void CellStore::listRefs() void CellStore::listRefs()
{ {
ESM::visit([&](auto&& cell) { listRefs(cell); }, mCellVariant); ESM::visit([&](auto&& cell) { listRefs(cell); }, mCellVariant);
@ -830,15 +860,7 @@ namespace MWWorld
void CellStore::loadRefs(const ESM4::Cell& cell, std::map<ESM::RefNum, ESM::RefId>& refNumToID) void CellStore::loadRefs(const ESM4::Cell& cell, std::map<ESM::RefNum, ESM::RefId>& refNumToID)
{ {
auto& refs = MWBase::Environment::get().getWorld()->getStore().get<ESM4::Reference>(); visitCell4References(cell, [&](ESM4::Reference& ref) { loadRef(ref, false); });
for (const auto& ref : refs)
{
if (ref.mParent == cell.mId)
{
loadRef(ref, false);
}
}
} }
void CellStore::loadRefs() void CellStore::loadRefs()

@ -29,7 +29,6 @@ namespace ESM4
class Reader; class Reader;
struct Static; struct Static;
struct Cell; struct Cell;
struct Reference;
struct Light; struct Light;
} }
@ -106,7 +105,7 @@ namespace MWWorld
// Special entry which is hardcoded and not loaded from an ESM // Special entry which is hardcoded and not loaded from an ESM
Store<ESM::Attribute>, Store<ESM::Attribute>,
Store<ESM4::Static>, Store<ESM4::Cell>, Store<ESM4::Reference>, Store<ESM4::Light>>; Store<ESM4::Static>, Store<ESM4::Cell>, Store<ESM4::Light>>;
private: private:
template <typename T> template <typename T>

@ -240,6 +240,8 @@ void ESM4::Cell::load(ESM4::Reader& reader)
throw std::runtime_error("ESM4::CELL::load - Unknown subrecord " + ESM::printName(subHdr.typeId)); throw std::runtime_error("ESM4::CELL::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
} }
} }
mReaderContext = reader.getContext();
} }
// void ESM4::Cell::save(ESM4::Writer& writer) const // void ESM4::Cell::save(ESM4::Writer& writer) const

@ -37,6 +37,7 @@
#include <components/esm/defs.hpp> #include <components/esm/defs.hpp>
#include <components/esm/refid.hpp> #include <components/esm/refid.hpp>
#include <components/esm3/cellid.hpp> #include <components/esm3/cellid.hpp>
#include <components/esm4/reader.hpp>
namespace ESM4 namespace ESM4
{ {
@ -96,6 +97,8 @@ namespace ESM4
CellGroup* mCellGroup; CellGroup* mCellGroup;
ESM4::ReaderContext mReaderContext;
void load(ESM4::Reader& reader); void load(ESM4::Reader& reader);
// void save(ESM4::Writer& writer) const; // void save(ESM4::Writer& writer) const;

Loading…
Cancel
Save