#include "load.hpp" #include "esmdata.hpp" #include "lessbyid.hpp" #include "record.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace EsmLoader { namespace { struct GetKey { template decltype(auto) operator()(const T& v) const { return (v.mId); } const ESM::CellId& operator()(const ESM::Cell& v) const { return v.mCellId; } std::pair operator()(const ESM::Land& v) const { return std::pair(v.mX, v.mY); } template decltype(auto) operator()(const Record& v) const { return (*this)(v.mValue); } }; struct CellRecords { Records mValues; std::map mByName; std::map, std::size_t> mByPosition; }; template > struct HasId : std::false_type {}; template struct HasId> : std::true_type {}; template constexpr bool hasId = HasId::value; template auto loadRecord(ESM::ESMReader& reader, Records& records) -> std::enable_if_t> { T record; bool deleted = false; record.load(reader, deleted); Misc::StringUtils::lowerCaseInPlace(record.mId); if (Misc::ResourceHelpers::isHiddenMarker(record.mId)) return; records.emplace_back(deleted, std::move(record)); } template auto loadRecord(ESM::ESMReader& reader, Records& records) -> std::enable_if_t> { T record; bool deleted = false; record.load(reader, deleted); records.emplace_back(deleted, std::move(record)); } void loadRecord(ESM::ESMReader& reader, CellRecords& records) { ESM::Cell record; bool deleted = false; record.loadNameAndData(reader, deleted); Misc::StringUtils::lowerCaseInPlace(record.mName); if ((record.mData.mFlags & ESM::Cell::Interior) != 0) { const auto it = records.mByName.find(record.mName); if (it == records.mByName.end()) { record.loadCell(reader, true); records.mByName.emplace_hint(it, record.mName, records.mValues.size()); records.mValues.emplace_back(deleted, std::move(record)); } else { Record& old = records.mValues[it->second]; old.mValue.mData = record.mData; old.mValue.loadCell(reader, true); } } else { const std::pair position(record.mData.mX, record.mData.mY); const auto it = records.mByPosition.find(position); if (it == records.mByPosition.end()) { record.loadCell(reader, true); records.mByPosition.emplace_hint(it, position, records.mValues.size()); records.mValues.emplace_back(deleted, std::move(record)); } else { Record& old = records.mValues[it->second]; old.mValue.mData = record.mData; old.mValue.loadCell(reader, true); } } } struct ShallowContent { Records mActivators; CellRecords mCells; Records mContainers; Records mDoors; Records mGameSettings; Records mLands; Records mStatics; }; void loadRecord(const Query& query, const ESM::NAME& name, ESM::ESMReader& reader, ShallowContent& content) { switch (name.toInt()) { case ESM::REC_ACTI: if (query.mLoadActivators) return loadRecord(reader, content.mActivators); break; case ESM::REC_CELL: if (query.mLoadCells) return loadRecord(reader, content.mCells); break; case ESM::REC_CONT: if (query.mLoadContainers) return loadRecord(reader, content.mContainers); break; case ESM::REC_DOOR: if (query.mLoadDoors) return loadRecord(reader, content.mDoors); break; case ESM::REC_GMST: if (query.mLoadGameSettings) return loadRecord(reader, content.mGameSettings); break; case ESM::REC_LAND: if (query.mLoadLands) return loadRecord(reader, content.mLands); break; case ESM::REC_STAT: if (query.mLoadStatics) return loadRecord(reader, content.mStatics); break; } reader.skipRecord(); } void loadEsm(const Query& query, ESM::ESMReader& reader, ShallowContent& content, Loading::Listener* listener) { Log(Debug::Info) << "Loading ESM file " << reader.getName(); while (reader.hasMoreRecs()) { const ESM::NAME recName = reader.getRecName(); reader.getRecHeader(); if (reader.getRecordFlags() & ESM::FLAG_Ignored) { reader.skipRecord(); continue; } loadRecord(query, recName, reader, content); if (listener != nullptr) listener->setProgress(fileProgress * reader.getFileOffset() / reader.getFileSize()); } } ShallowContent shallowLoad(const Query& query, const std::vector& contentFiles, const Files::Collections& fileCollections, ESM::ReadersCache& readers, ToUTF8::Utf8Encoder* encoder, Loading::Listener* listener) { ShallowContent result; const std::set supportedFormats { ".esm", ".esp", ".omwgame", ".omwaddon", ".project", }; for (std::size_t i = 0; i < contentFiles.size(); ++i) { const std::string &file = contentFiles[i]; const std::string extension = Misc::StringUtils::lowerCase(Files::pathToUnicodeString(std::filesystem::path(file).extension())); if (supportedFormats.find(extension) == supportedFormats.end()) { Log(Debug::Warning) << "Skipping unsupported content file: " << file; continue; } if (listener != nullptr) { listener->setLabel(file); listener->setProgressRange(fileProgress); } const Files::MultiDirCollection& collection = fileCollections.getCollection(extension); const ESM::ReadersCache::BusyItem reader = readers.get(i); reader->setEncoder(encoder); reader->setIndex(static_cast(i)); reader->open(collection.getPath(file)); if (query.mLoadCells) reader->resolveParentFileIndices(readers); loadEsm(query, *reader, result, listener); } return result; } struct WithType { ESM::RecNameInts mType; template RefIdWithType operator()(const T& v) const { return {v.mId, mType}; } }; template void addRefIdsTypes(const std::vector& values, std::vector& refIdsTypes) { std::transform(values.begin(), values.end(), std::back_inserter(refIdsTypes), WithType {static_cast(T::sRecordId)}); } void addRefIdsTypes(EsmData& content) { content.mRefIdTypes.reserve( content.mActivators.size() + content.mContainers.size() + content.mDoors.size() + content.mStatics.size() ); addRefIdsTypes(content.mActivators, content.mRefIdTypes); addRefIdsTypes(content.mContainers, content.mRefIdTypes); addRefIdsTypes(content.mDoors, content.mRefIdTypes); addRefIdsTypes(content.mStatics, content.mRefIdTypes); std::sort(content.mRefIdTypes.begin(), content.mRefIdTypes.end(), LessById {}); } std::vector prepareCellRecords(Records& records) { std::vector result; for (Record& v : records) if (!v.mDeleted) result.emplace_back(std::move(v.mValue)); return result; } } EsmData loadEsmData(const Query& query, const std::vector& contentFiles, const Files::Collections& fileCollections, ESM::ReadersCache& readers, ToUTF8::Utf8Encoder* encoder, Loading::Listener* listener) { Log(Debug::Info) << "Loading ESM data..."; ShallowContent content = shallowLoad(query, contentFiles, fileCollections, readers, encoder, listener); std::ostringstream loaded; if (query.mLoadActivators) loaded << ' ' << content.mActivators.size() << " activators,"; if (query.mLoadCells) loaded << ' ' << content.mCells.mValues.size() << " cells,"; if (query.mLoadContainers) loaded << ' ' << content.mContainers.size() << " containers,"; if (query.mLoadDoors) loaded << ' ' << content.mDoors.size() << " doors,"; if (query.mLoadGameSettings) loaded << ' ' << content.mGameSettings.size() << " game settings,"; if (query.mLoadLands) loaded << ' ' << content.mLands.size() << " lands,"; if (query.mLoadStatics) loaded << ' ' << content.mStatics.size() << " statics,"; Log(Debug::Info) << "Loaded" << loaded.str(); EsmData result; if (query.mLoadActivators) result.mActivators = prepareRecords(content.mActivators, GetKey {}); if (query.mLoadCells) result.mCells = prepareCellRecords(content.mCells.mValues); if (query.mLoadContainers) result.mContainers = prepareRecords(content.mContainers, GetKey {}); if (query.mLoadDoors) result.mDoors = prepareRecords(content.mDoors, GetKey {}); if (query.mLoadGameSettings) result.mGameSettings = prepareRecords(content.mGameSettings, GetKey {}); if (query.mLoadLands) result.mLands = prepareRecords(content.mLands, GetKey {}); if (query.mLoadStatics) result.mStatics = prepareRecords(content.mStatics, GetKey {}); addRefIdsTypes(result); std::ostringstream prepared; if (query.mLoadActivators) prepared << ' ' << result.mActivators.size() << " unique activators,"; if (query.mLoadCells) prepared << ' ' << result.mCells.size() << " unique cells,"; if (query.mLoadContainers) prepared << ' ' << result.mContainers.size() << " unique containers,"; if (query.mLoadDoors) prepared << ' ' << result.mDoors.size() << " unique doors,"; if (query.mLoadGameSettings) prepared << ' ' << result.mGameSettings.size() << " unique game settings,"; if (query.mLoadLands) prepared << ' ' << result.mLands.size() << " unique lands,"; if (query.mLoadStatics) prepared << ' ' << result.mStatics.size() << " unique statics,"; Log(Debug::Info) << "Prepared" << prepared.str(); return result; } }