2013-05-06 12:11:55 +00:00
|
|
|
#ifndef CSM_WOLRD_REFIDDATA_H
|
|
|
|
#define CSM_WOLRD_REFIDDATA_H
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
2021-07-23 04:21:21 +00:00
|
|
|
#include <memory>
|
2021-07-24 11:23:03 +00:00
|
|
|
#include <cassert>
|
2021-09-04 16:07:23 +00:00
|
|
|
#include <string_view>
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2022-01-22 14:58:41 +00:00
|
|
|
#include <components/esm3/loadacti.hpp>
|
|
|
|
#include <components/esm3/loadalch.hpp>
|
|
|
|
#include <components/esm3/loadappa.hpp>
|
|
|
|
#include <components/esm3/loadarmo.hpp>
|
|
|
|
#include <components/esm3/loadbook.hpp>
|
|
|
|
#include <components/esm3/loadclot.hpp>
|
|
|
|
#include <components/esm3/loadcont.hpp>
|
|
|
|
#include <components/esm3/loadcrea.hpp>
|
|
|
|
#include <components/esm3/loaddoor.hpp>
|
|
|
|
#include <components/esm3/loadingr.hpp>
|
|
|
|
#include <components/esm3/loadlevlist.hpp>
|
|
|
|
#include <components/esm3/loadligh.hpp>
|
|
|
|
#include <components/esm3/loadlock.hpp>
|
|
|
|
#include <components/esm3/loadprob.hpp>
|
|
|
|
#include <components/esm3/loadrepa.hpp>
|
|
|
|
#include <components/esm3/loadstat.hpp>
|
|
|
|
#include <components/esm3/loadweap.hpp>
|
|
|
|
#include <components/esm3/loadnpc.hpp>
|
|
|
|
#include <components/esm3/loadmisc.hpp>
|
|
|
|
#include <components/esm3/esmwriter.hpp>
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2022-08-02 22:00:54 +00:00
|
|
|
#include <components/misc/strings/algorithm.hpp>
|
2015-07-14 13:18:33 +00:00
|
|
|
|
2013-05-06 12:11:55 +00:00
|
|
|
#include "record.hpp"
|
|
|
|
#include "universalid.hpp"
|
|
|
|
|
|
|
|
namespace ESM
|
|
|
|
{
|
|
|
|
class ESMReader;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace CSMWorld
|
|
|
|
{
|
|
|
|
struct RefIdDataContainerBase
|
|
|
|
{
|
|
|
|
virtual ~RefIdDataContainerBase();
|
|
|
|
|
|
|
|
virtual int getSize() const = 0;
|
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
virtual const RecordBase& getRecord (int index) const = 0;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
virtual RecordBase& getRecord (int index)= 0;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2021-06-30 00:09:56 +00:00
|
|
|
virtual unsigned int getRecordFlags (int index) const = 0;
|
|
|
|
|
2014-02-15 14:50:17 +00:00
|
|
|
virtual void appendRecord (const std::string& id, bool base) = 0;
|
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
virtual void insertRecord (std::unique_ptr<RecordBase> record) = 0;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2015-07-14 13:18:33 +00:00
|
|
|
virtual int load (ESM::ESMReader& reader, bool base) = 0;
|
|
|
|
///< \return index of a loaded record or -1 if no record was loaded
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
virtual void erase (int index, int count) = 0;
|
2015-04-02 09:19:15 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
virtual std::string getId (int index) const = 0;
|
2013-09-24 11:53:19 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
virtual void save (int index, ESM::ESMWriter& writer) const = 0;
|
2013-05-06 12:11:55 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename RecordT>
|
|
|
|
struct RefIdDataContainer : public RefIdDataContainerBase
|
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
std::vector<std::unique_ptr<Record<RecordT> > > mContainer;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
int getSize() const override;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
const RecordBase& getRecord (int index) const override;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
RecordBase& getRecord (int index) override;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2021-06-30 00:09:56 +00:00
|
|
|
unsigned int getRecordFlags (int index) const override;
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void appendRecord (const std::string& id, bool base) override;
|
2014-02-15 14:50:17 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
void insertRecord (std::unique_ptr<RecordBase> record) override;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
int load (ESM::ESMReader& reader, bool base) override;
|
2015-07-14 13:18:33 +00:00
|
|
|
///< \return index of a loaded record or -1 if no record was loaded
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void erase (int index, int count) override;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
std::string getId (int index) const override;
|
2013-09-24 11:53:19 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void save (int index, ESM::ESMWriter& writer) const override;
|
2013-05-06 12:11:55 +00:00
|
|
|
};
|
|
|
|
|
2014-01-21 08:43:02 +00:00
|
|
|
template<typename RecordT>
|
2021-07-23 04:21:21 +00:00
|
|
|
void RefIdDataContainer<RecordT>::insertRecord(std::unique_ptr<RecordBase> record)
|
2014-01-21 08:43:02 +00:00
|
|
|
{
|
2021-07-24 11:23:03 +00:00
|
|
|
assert(record != nullptr);
|
|
|
|
// convert base pointer to record type pointer
|
|
|
|
std::unique_ptr<Record<RecordT>> typedRecord(&dynamic_cast<Record<RecordT>&>(*record));
|
|
|
|
record.release();
|
|
|
|
mContainer.push_back(std::move(typedRecord));
|
2014-01-21 08:43:02 +00:00
|
|
|
}
|
2014-02-15 14:50:17 +00:00
|
|
|
|
2013-05-06 12:11:55 +00:00
|
|
|
template<typename RecordT>
|
|
|
|
int RefIdDataContainer<RecordT>::getSize() const
|
|
|
|
{
|
2013-12-30 17:47:19 +00:00
|
|
|
return static_cast<int> (mContainer.size());
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RecordT>
|
2013-12-30 17:47:19 +00:00
|
|
|
const RecordBase& RefIdDataContainer<RecordT>::getRecord (int index) const
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
return *mContainer.at (index);
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RecordT>
|
2013-12-30 17:47:19 +00:00
|
|
|
RecordBase& RefIdDataContainer<RecordT>::getRecord (int index)
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
return *mContainer.at (index);
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
2021-06-30 00:09:56 +00:00
|
|
|
template<typename RecordT>
|
|
|
|
unsigned int RefIdDataContainer<RecordT>::getRecordFlags (int index) const
|
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
return mContainer.at (index)->get().mRecordFlags;
|
2021-06-30 00:09:56 +00:00
|
|
|
}
|
|
|
|
|
2013-05-06 12:11:55 +00:00
|
|
|
template<typename RecordT>
|
2014-02-15 14:50:17 +00:00
|
|
|
void RefIdDataContainer<RecordT>::appendRecord (const std::string& id, bool base)
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record = std::make_unique<Record<RecordT>>();
|
2014-02-15 14:50:17 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_ModifiedOnly;
|
2014-02-15 14:50:17 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mBase.mId = id;
|
|
|
|
record->mModified.mId = id;
|
|
|
|
(base ? record->mBase : record->mModified).blank();
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer.push_back (std::move(record));
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RecordT>
|
2015-07-14 13:18:33 +00:00
|
|
|
int RefIdDataContainer<RecordT>::load (ESM::ESMReader& reader, bool base)
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2015-07-12 12:22:51 +00:00
|
|
|
RecordT record;
|
2015-07-21 17:25:43 +00:00
|
|
|
bool isDeleted = false;
|
|
|
|
|
|
|
|
record.load(reader, isDeleted);
|
2015-07-12 12:22:51 +00:00
|
|
|
|
2015-07-14 13:18:33 +00:00
|
|
|
int index = 0;
|
|
|
|
int numRecords = static_cast<int>(mContainer.size());
|
|
|
|
for (; index < numRecords; ++index)
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(mContainer[index]->get().mId, record.mId))
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-21 17:25:43 +00:00
|
|
|
if (isDeleted)
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
2015-07-14 13:18:33 +00:00
|
|
|
if (index == numRecords)
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
|
|
|
// deleting a record that does not exist
|
|
|
|
// ignore it for now
|
|
|
|
/// \todo report the problem to the user
|
2015-07-14 13:18:33 +00:00
|
|
|
return -1;
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
|
|
|
|
2015-07-14 13:18:33 +00:00
|
|
|
// Flag the record as Deleted even for a base content file.
|
|
|
|
// RefIdData is responsible for its erasure.
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer[index]->mState = RecordBase::State_Deleted;
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-07-14 13:18:33 +00:00
|
|
|
if (index == numRecords)
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
|
|
|
appendRecord(record.mId, base);
|
|
|
|
if (base)
|
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer.back()->mBase = record;
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer.back()->mModified = record;
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
|
|
|
}
|
2015-07-14 13:18:33 +00:00
|
|
|
else if (!base)
|
2015-07-12 12:22:51 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer[index]->setModified(record);
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
2016-09-24 20:45:08 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Overwrite
|
2021-07-23 04:21:21 +00:00
|
|
|
mContainer[index]->setModified(record);
|
|
|
|
mContainer[index]->merge();
|
2016-09-24 20:45:08 +00:00
|
|
|
}
|
2015-07-12 12:22:51 +00:00
|
|
|
}
|
2015-07-14 13:18:33 +00:00
|
|
|
|
|
|
|
return index;
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RecordT>
|
2013-12-30 17:47:19 +00:00
|
|
|
void RefIdDataContainer<RecordT>::erase (int index, int count)
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2015-04-24 20:04:39 +00:00
|
|
|
if (index<0 || index+count>getSize())
|
2013-12-30 17:47:19 +00:00
|
|
|
throw std::runtime_error ("invalid RefIdDataContainer index");
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
mContainer.erase (mContainer.begin()+index, mContainer.begin()+index+count);
|
2015-04-02 09:19:15 +00:00
|
|
|
}
|
2013-05-06 12:11:55 +00:00
|
|
|
|
|
|
|
template<typename RecordT>
|
2013-12-30 17:47:19 +00:00
|
|
|
std::string RefIdDataContainer<RecordT>::getId (int index) const
|
2013-05-06 12:11:55 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
return mContainer.at (index)->get().mId;
|
2013-05-06 12:11:55 +00:00
|
|
|
}
|
|
|
|
|
2013-09-24 11:53:19 +00:00
|
|
|
template<typename RecordT>
|
2013-12-30 17:47:19 +00:00
|
|
|
void RefIdDataContainer<RecordT>::save (int index, ESM::ESMWriter& writer) const
|
2013-09-24 11:53:19 +00:00
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
const Record<RecordT>& record = *mContainer.at(index);
|
2013-09-24 11:53:19 +00:00
|
|
|
|
2015-07-14 13:18:33 +00:00
|
|
|
if (record.isModified() || record.mState == RecordBase::State_Deleted)
|
2013-09-24 11:53:19 +00:00
|
|
|
{
|
2015-07-21 17:25:43 +00:00
|
|
|
RecordT esmRecord = record.get();
|
2021-06-29 13:25:26 +00:00
|
|
|
writer.startRecord(esmRecord.sRecordId, esmRecord.mRecordFlags);
|
2015-07-21 17:25:43 +00:00
|
|
|
esmRecord.save(writer, record.mState == RecordBase::State_Deleted);
|
2015-07-14 13:18:33 +00:00
|
|
|
writer.endRecord(esmRecord.sRecordId);
|
2013-09-24 11:53:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-06 12:11:55 +00:00
|
|
|
class RefIdData
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
typedef std::pair<int, UniversalId::Type> LocalIndex;
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
RefIdDataContainer<ESM::Activator> mActivators;
|
|
|
|
RefIdDataContainer<ESM::Potion> mPotions;
|
|
|
|
RefIdDataContainer<ESM::Apparatus> mApparati;
|
|
|
|
RefIdDataContainer<ESM::Armor> mArmors;
|
|
|
|
RefIdDataContainer<ESM::Book> mBooks;
|
|
|
|
RefIdDataContainer<ESM::Clothing> mClothing;
|
|
|
|
RefIdDataContainer<ESM::Container> mContainers;
|
|
|
|
RefIdDataContainer<ESM::Creature> mCreatures;
|
|
|
|
RefIdDataContainer<ESM::Door> mDoors;
|
|
|
|
RefIdDataContainer<ESM::Ingredient> mIngredients;
|
|
|
|
RefIdDataContainer<ESM::CreatureLevList> mCreatureLevelledLists;
|
|
|
|
RefIdDataContainer<ESM::ItemLevList> mItemLevelledLists;
|
|
|
|
RefIdDataContainer<ESM::Light> mLights;
|
|
|
|
RefIdDataContainer<ESM::Lockpick> mLockpicks;
|
|
|
|
RefIdDataContainer<ESM::Miscellaneous> mMiscellaneous;
|
|
|
|
RefIdDataContainer<ESM::NPC> mNpcs;
|
|
|
|
RefIdDataContainer<ESM::Probe> mProbes;
|
|
|
|
RefIdDataContainer<ESM::Repair> mRepairs;
|
|
|
|
RefIdDataContainer<ESM::Static> mStatics;
|
|
|
|
RefIdDataContainer<ESM::Weapon> mWeapons;
|
|
|
|
|
|
|
|
std::map<std::string, LocalIndex> mIndex;
|
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
std::map<UniversalId::Type, RefIdDataContainerBase *> mRecordContainers;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
void erase (const LocalIndex& index, int count);
|
2013-05-06 12:11:55 +00:00
|
|
|
///< Must not spill over into another type.
|
|
|
|
|
2015-07-14 13:18:33 +00:00
|
|
|
std::string getRecordId(const LocalIndex &index) const;
|
|
|
|
|
2013-05-06 12:11:55 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
RefIdData();
|
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
LocalIndex globalToLocalIndex (int index) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
int localToGlobalIndex (const LocalIndex& index) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2021-09-04 16:07:23 +00:00
|
|
|
LocalIndex searchId(std::string_view id) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
void erase (int index, int count);
|
2014-02-15 14:50:17 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
void insertRecord (std::unique_ptr<RecordBase> record, CSMWorld::UniversalId::Type type,
|
2015-08-25 10:40:40 +00:00
|
|
|
const std::string& id);
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
const RecordBase& getRecord (const LocalIndex& index) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
RecordBase& getRecord (const LocalIndex& index);
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2021-06-29 13:25:26 +00:00
|
|
|
unsigned int getRecordFlags(const std::string& id) const;
|
|
|
|
|
2014-02-15 14:50:17 +00:00
|
|
|
void appendRecord (UniversalId::Type type, const std::string& id, bool base);
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
int getAppendIndex (UniversalId::Type type) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
|
2015-07-12 12:22:51 +00:00
|
|
|
void load (ESM::ESMReader& reader, bool base, UniversalId::Type type);
|
2013-05-06 12:11:55 +00:00
|
|
|
|
|
|
|
int getSize() const;
|
2013-09-19 10:11:27 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
std::vector<std::string> getIds (bool listDeleted = true) const;
|
2013-09-19 10:11:27 +00:00
|
|
|
///< Return a sorted collection of all IDs
|
|
|
|
///
|
2013-09-19 11:42:19 +00:00
|
|
|
/// \param listDeleted include deleted record in the list
|
2013-09-24 11:53:19 +00:00
|
|
|
|
2013-12-30 17:47:19 +00:00
|
|
|
void save (int index, ESM::ESMWriter& writer) const;
|
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
//RECORD CONTAINERS ACCESS METHODS
|
2014-05-25 13:46:23 +00:00
|
|
|
const RefIdDataContainer<ESM::Book>& getBooks() const;
|
|
|
|
const RefIdDataContainer<ESM::Activator>& getActivators() const;
|
|
|
|
const RefIdDataContainer<ESM::Potion>& getPotions() const;
|
|
|
|
const RefIdDataContainer<ESM::Apparatus>& getApparati() const;
|
|
|
|
const RefIdDataContainer<ESM::Armor>& getArmors() const;
|
|
|
|
const RefIdDataContainer<ESM::Clothing>& getClothing() const;
|
|
|
|
const RefIdDataContainer<ESM::Container>& getContainers() const;
|
|
|
|
const RefIdDataContainer<ESM::Creature>& getCreatures() const;
|
|
|
|
const RefIdDataContainer<ESM::Door>& getDoors() const;
|
|
|
|
const RefIdDataContainer<ESM::Ingredient>& getIngredients() const;
|
|
|
|
const RefIdDataContainer<ESM::CreatureLevList>& getCreatureLevelledLists() const;
|
|
|
|
const RefIdDataContainer<ESM::ItemLevList>& getItemLevelledList() const;
|
|
|
|
const RefIdDataContainer<ESM::Light>& getLights() const;
|
|
|
|
const RefIdDataContainer<ESM::Lockpick>& getLocpicks() const;
|
|
|
|
const RefIdDataContainer<ESM::Miscellaneous>& getMiscellaneous() const;
|
|
|
|
const RefIdDataContainer<ESM::NPC>& getNPCs() const;
|
|
|
|
const RefIdDataContainer<ESM::Weapon >& getWeapons() const;
|
|
|
|
const RefIdDataContainer<ESM::Probe >& getProbes() const;
|
|
|
|
const RefIdDataContainer<ESM::Repair>& getRepairs() const;
|
|
|
|
const RefIdDataContainer<ESM::Static>& getStatics() const;
|
2015-08-25 09:54:16 +00:00
|
|
|
|
|
|
|
void copyTo (int index, RefIdData& target) const;
|
2013-05-06 12:11:55 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|