mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 06:35:30 +00:00
aa9a99bde5
Fix magic effects in the editor Closes #7481 See merge request OpenMW/openmw!3798
627 lines
20 KiB
C++
627 lines
20 KiB
C++
#ifndef CSM_WOLRD_COLLECTION_H
|
|
#define CSM_WOLRD_COLLECTION_H
|
|
|
|
#include <algorithm>
|
|
#include <cctype>
|
|
#include <functional>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <stdexcept>
|
|
#include <string>
|
|
#include <string_view>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#include <QVariant>
|
|
|
|
#include <components/esm3/loaddial.hpp>
|
|
#include <components/esm3/loadmgef.hpp>
|
|
#include <components/esm3/loadskil.hpp>
|
|
#include <components/misc/strings/lower.hpp>
|
|
|
|
#include "collectionbase.hpp"
|
|
#include "columnbase.hpp"
|
|
#include "columnimp.hpp"
|
|
#include "info.hpp"
|
|
#include "land.hpp"
|
|
#include "landtexture.hpp"
|
|
#include "record.hpp"
|
|
#include "ref.hpp"
|
|
|
|
namespace CSMWorld
|
|
{
|
|
inline std::pair<std::string_view, std::string_view> parseInfoRefId(const ESM::RefId& infoId)
|
|
{
|
|
const auto separator = infoId.getRefIdString().find('#');
|
|
if (separator == std::string::npos)
|
|
throw std::runtime_error("Invalid info id: " + infoId.getRefIdString());
|
|
const std::string_view view(infoId.getRefIdString());
|
|
return { view.substr(0, separator), view.substr(separator + 1) };
|
|
}
|
|
|
|
template <typename T>
|
|
void setRecordId(const decltype(T::mId)& id, T& record)
|
|
{
|
|
record.mId = id;
|
|
}
|
|
|
|
inline void setRecordId(const ESM::RefId& id, Info& record)
|
|
{
|
|
record.mId = id;
|
|
const auto [topicId, originalId] = parseInfoRefId(id);
|
|
record.mTopicId = ESM::RefId::stringRefId(topicId);
|
|
record.mOriginalId = ESM::RefId::stringRefId(originalId);
|
|
}
|
|
|
|
template <typename T>
|
|
auto getRecordId(const T& record)
|
|
{
|
|
return record.mId;
|
|
}
|
|
|
|
inline void setRecordId(const ESM::RefId& id, Land& record)
|
|
{
|
|
int x = 0;
|
|
int y = 0;
|
|
|
|
Land::parseUniqueRecordId(id.getRefIdString(), x, y);
|
|
record.mX = x;
|
|
record.mY = y;
|
|
}
|
|
|
|
inline ESM::RefId getRecordId(const Land& record)
|
|
{
|
|
return ESM::RefId::stringRefId(Land::createUniqueRecordId(record.mX, record.mY));
|
|
}
|
|
|
|
inline void setRecordId(const ESM::RefId& id, LandTexture& record)
|
|
{
|
|
int plugin = 0;
|
|
int index = 0;
|
|
|
|
LandTexture::parseUniqueRecordId(id.getRefIdString(), plugin, index);
|
|
record.mPluginIndex = plugin;
|
|
record.mIndex = index;
|
|
}
|
|
|
|
inline ESM::RefId getRecordId(const ESM::MagicEffect& record)
|
|
{
|
|
return ESM::RefId::stringRefId(CSMWorld::getStringId(record.mId));
|
|
}
|
|
|
|
inline void setRecordId(const ESM::RefId& id, ESM::MagicEffect& record)
|
|
{
|
|
int index = ESM::MagicEffect::indexNameToIndex(id.getRefIdString());
|
|
record.mId = ESM::RefId::index(ESM::REC_MGEF, static_cast<std::uint32_t>(index));
|
|
}
|
|
|
|
inline ESM::RefId getRecordId(const LandTexture& record)
|
|
{
|
|
return ESM::RefId::stringRefId(LandTexture::createUniqueRecordId(record.mPluginIndex, record.mIndex));
|
|
}
|
|
|
|
inline void setRecordId(const ESM::RefId& id, ESM::Skill& record)
|
|
{
|
|
if (const auto* skillId = id.getIf<ESM::SkillId>())
|
|
record.mId = *skillId;
|
|
throw std::runtime_error("Invalid skill id: " + id.toDebugString());
|
|
}
|
|
|
|
/// \brief Single-type record collection
|
|
template <typename ESXRecordT>
|
|
class Collection : public CollectionBase
|
|
{
|
|
public:
|
|
typedef ESXRecordT ESXRecord;
|
|
|
|
private:
|
|
std::vector<std::unique_ptr<Record<ESXRecordT>>> mRecords;
|
|
std::map<ESM::RefId, int> mIndex;
|
|
std::vector<Column<ESXRecordT>*> mColumns;
|
|
|
|
protected:
|
|
const std::vector<std::unique_ptr<Record<ESXRecordT>>>& getRecords() const;
|
|
|
|
void reorderRowsImp(const std::vector<int>& indexOrder);
|
|
|
|
bool reorderRowsImp(int baseIndex, const std::vector<int>& newOrder);
|
|
///< Reorder the rows [baseIndex, baseIndex+newOrder.size()) according to the indices
|
|
/// given in \a newOrder (baseIndex+newOrder[0] specifies the new index of row baseIndex).
|
|
///
|
|
/// \return Success?
|
|
|
|
int cloneRecordImp(const ESM::RefId& origin, const ESM::RefId& dest, UniversalId::Type type);
|
|
///< Returns the index of the clone.
|
|
|
|
int touchRecordImp(const ESM::RefId& id);
|
|
///< Returns the index of the record on success, -1 on failure.
|
|
|
|
public:
|
|
Collection() = default;
|
|
Collection(const Collection&) = delete;
|
|
Collection& operator=(const Collection&) = delete;
|
|
|
|
~Collection() override;
|
|
|
|
void add(const ESXRecordT& record);
|
|
///< Add a new record (modified)
|
|
|
|
int getSize() const override;
|
|
|
|
ESM::RefId getId(int index) const override;
|
|
|
|
int getIndex(const ESM::RefId& id) const override;
|
|
|
|
int getColumns() const override;
|
|
|
|
QVariant getData(int index, int column) const override;
|
|
|
|
void setData(int index, int column, const QVariant& data) override;
|
|
|
|
const ColumnBase& getColumn(int column) const override;
|
|
|
|
void merge();
|
|
///< Merge modified into base.
|
|
|
|
void purge();
|
|
///< Remove records that are flagged as erased.
|
|
|
|
void removeRows(int index, int count) override;
|
|
|
|
void appendBlankRecord(const ESM::RefId& id, UniversalId::Type type = UniversalId::Type_None) override;
|
|
///< \param type Will be ignored, unless the collection supports multiple record types
|
|
|
|
void cloneRecord(
|
|
const ESM::RefId& origin, const ESM::RefId& destination, const UniversalId::Type type) override;
|
|
|
|
bool touchRecord(const ESM::RefId& id) override;
|
|
///< Change the state of a record from base to modified, if it is not already.
|
|
/// \return True if the record was changed.
|
|
|
|
int searchId(const ESM::RefId& id) const override;
|
|
////< Search record with \a id.
|
|
/// \return index of record (if found) or -1 (not found)
|
|
|
|
void replace(int index, std::unique_ptr<RecordBase> record) override;
|
|
///< If the record type does not match, an exception is thrown.
|
|
///
|
|
/// \attention \a record must not change the ID.
|
|
|
|
void appendRecord(std::unique_ptr<RecordBase> record, UniversalId::Type type = UniversalId::Type_None) override;
|
|
///< If the record type does not match, an exception is thrown.
|
|
///< \param type Will be ignored, unless the collection supports multiple record types
|
|
|
|
const Record<ESXRecordT>& getRecord(const ESM::RefId& id) const override;
|
|
|
|
const Record<ESXRecordT>& getRecord(int index) const override;
|
|
|
|
int getAppendIndex(const ESM::RefId& id, UniversalId::Type type = UniversalId::Type_None) const override;
|
|
///< \param type Will be ignored, unless the collection supports multiple record types
|
|
|
|
std::vector<ESM::RefId> getIds(bool listDeleted = true) const override;
|
|
///< Return a sorted collection of all IDs
|
|
///
|
|
/// \param listDeleted include deleted record in the list
|
|
|
|
virtual void insertRecord(
|
|
std::unique_ptr<RecordBase> record, int index, UniversalId::Type type = UniversalId::Type_None);
|
|
///< Insert record before index.
|
|
///
|
|
/// If the record type does not match, an exception is thrown.
|
|
///
|
|
/// If the index is invalid either generally (by being out of range) or for the particular
|
|
/// record, an exception is thrown.
|
|
|
|
bool reorderRows(int baseIndex, const std::vector<int>& newOrder) override;
|
|
///< Reorder the rows [baseIndex, baseIndex+newOrder.size()) according to the indices
|
|
/// given in \a newOrder (baseIndex+newOrder[0] specifies the new index of row baseIndex).
|
|
///
|
|
/// \return Success?
|
|
|
|
void addColumn(Column<ESXRecordT>* column);
|
|
|
|
void setRecord(int index, std::unique_ptr<Record<ESXRecordT>> record);
|
|
///< \attention This function must not change the ID.
|
|
|
|
NestableColumn* getNestableColumn(int column) const;
|
|
};
|
|
|
|
template <typename ESXRecordT>
|
|
const std::vector<std::unique_ptr<Record<ESXRecordT>>>& Collection<ESXRecordT>::getRecords() const
|
|
{
|
|
return mRecords;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::reorderRowsImp(const std::vector<int>& indexOrder)
|
|
{
|
|
assert(indexOrder.size() == mRecords.size());
|
|
assert(std::unordered_set(indexOrder.begin(), indexOrder.end()).size() == indexOrder.size());
|
|
std::vector<std::unique_ptr<Record<ESXRecordT>>> orderedRecords;
|
|
for (const int index : indexOrder)
|
|
{
|
|
mIndex.at(mRecords[index]->get().mId) = static_cast<int>(orderedRecords.size());
|
|
orderedRecords.push_back(std::move(mRecords[index]));
|
|
}
|
|
mRecords = std::move(orderedRecords);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
bool Collection<ESXRecordT>::reorderRowsImp(int baseIndex, const std::vector<int>& newOrder)
|
|
{
|
|
if (!newOrder.empty())
|
|
{
|
|
int size = static_cast<int>(newOrder.size());
|
|
|
|
// check that all indices are present
|
|
std::vector<int> test(newOrder);
|
|
std::sort(test.begin(), test.end());
|
|
if (*test.begin() != 0 || *--test.end() != size - 1)
|
|
return false;
|
|
|
|
// reorder records
|
|
std::vector<std::unique_ptr<Record<ESXRecordT>>> buffer(size);
|
|
|
|
for (int i = 0; i < size; ++i)
|
|
{
|
|
buffer[newOrder[i]] = std::move(mRecords[baseIndex + i]);
|
|
if (buffer[newOrder[i]])
|
|
buffer[newOrder[i]]->setModified(buffer[newOrder[i]]->get());
|
|
}
|
|
|
|
std::move(buffer.begin(), buffer.end(), mRecords.begin() + baseIndex);
|
|
|
|
// adjust index
|
|
for (auto& [id, index] : mIndex)
|
|
if (index >= baseIndex && index < baseIndex + size)
|
|
index = newOrder.at(index - baseIndex) + baseIndex;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::cloneRecordImp(
|
|
const ESM::RefId& origin, const ESM::RefId& destination, UniversalId::Type type)
|
|
{
|
|
auto copy = std::make_unique<Record<ESXRecordT>>();
|
|
copy->mModified = getRecord(origin).get();
|
|
copy->mState = RecordBase::State_ModifiedOnly;
|
|
setRecordId(destination, copy->get());
|
|
|
|
if constexpr (std::is_same_v<ESXRecordT, CSMWorld::CellRef>)
|
|
{
|
|
if (type == UniversalId::Type_Reference)
|
|
{
|
|
CSMWorld::CellRef* ptr = (CSMWorld::CellRef*)©->mModified;
|
|
ptr->mRefNum.mIndex = 0;
|
|
}
|
|
}
|
|
|
|
if constexpr (std::is_same_v<ESXRecordT, ESM::Dialogue>)
|
|
{
|
|
copy->mModified.mStringId = copy->mModified.mId.getRefIdString();
|
|
}
|
|
|
|
const int index = getAppendIndex(destination, type);
|
|
insertRecord(std::move(copy), getAppendIndex(destination, type));
|
|
|
|
return index;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::touchRecordImp(const ESM::RefId& id)
|
|
{
|
|
const int index = getIndex(id);
|
|
Record<ESXRecordT>& record = *mRecords.at(index);
|
|
if (record.isDeleted())
|
|
throw std::runtime_error("attempt to touch deleted record from collection of "
|
|
+ std::string(ESXRecordT::getRecordType()) + ": " + id.toDebugString());
|
|
|
|
if (!record.isModified())
|
|
{
|
|
record.setModified(record.get());
|
|
return index;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::cloneRecord(
|
|
const ESM::RefId& origin, const ESM::RefId& destination, const UniversalId::Type type)
|
|
{
|
|
cloneRecordImp(origin, destination, type);
|
|
}
|
|
|
|
template <>
|
|
inline void Collection<Land>::cloneRecord(
|
|
const ESM::RefId& origin, const ESM::RefId& destination, const UniversalId::Type type)
|
|
{
|
|
const int index = cloneRecordImp(origin, destination, type);
|
|
mRecords.at(index)->get().setPlugin(0);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
bool Collection<ESXRecordT>::touchRecord(const ESM::RefId& id)
|
|
{
|
|
return touchRecordImp(id) != -1;
|
|
}
|
|
|
|
template <>
|
|
inline bool Collection<Land>::touchRecord(const ESM::RefId& id)
|
|
{
|
|
const int index = touchRecordImp(id);
|
|
if (index >= 0)
|
|
{
|
|
mRecords.at(index)->get().setPlugin(0);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
Collection<ESXRecordT>::~Collection()
|
|
{
|
|
for (typename std::vector<Column<ESXRecordT>*>::iterator iter(mColumns.begin()); iter != mColumns.end(); ++iter)
|
|
delete *iter;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::add(const ESXRecordT& record)
|
|
{
|
|
const ESM::RefId id = getRecordId(record);
|
|
|
|
auto iter = mIndex.find(id);
|
|
|
|
if (iter == mIndex.end())
|
|
{
|
|
auto record2 = std::make_unique<Record<ESXRecordT>>();
|
|
record2->mState = Record<ESXRecordT>::State_ModifiedOnly;
|
|
record2->mModified = record;
|
|
|
|
insertRecord(std::move(record2), getAppendIndex(id));
|
|
}
|
|
else
|
|
{
|
|
mRecords[iter->second]->setModified(record);
|
|
}
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::getSize() const
|
|
{
|
|
return mRecords.size();
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
ESM::RefId Collection<ESXRecordT>::getId(int index) const
|
|
{
|
|
return getRecordId(mRecords.at(index)->get());
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::getIndex(const ESM::RefId& id) const
|
|
{
|
|
int index = searchId(id);
|
|
|
|
if (index == -1)
|
|
throw std::runtime_error("ID is not found in collection of " + std::string(ESXRecordT::getRecordType())
|
|
+ " records: " + id.getRefIdString());
|
|
|
|
return index;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::getColumns() const
|
|
{
|
|
return mColumns.size();
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
QVariant Collection<ESXRecordT>::getData(int index, int column) const
|
|
{
|
|
return mColumns.at(column)->get(*mRecords.at(index));
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::setData(int index, int column, const QVariant& data)
|
|
{
|
|
return mColumns.at(column)->set(*mRecords.at(index), data);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
const ColumnBase& Collection<ESXRecordT>::getColumn(int column) const
|
|
{
|
|
return *mColumns.at(column);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
NestableColumn* Collection<ESXRecordT>::getNestableColumn(int column) const
|
|
{
|
|
if (column < 0 || column >= static_cast<int>(mColumns.size()))
|
|
throw std::runtime_error(
|
|
"column index out of range [0, " + std::to_string(mColumns.size()) + "): " + std::to_string(column));
|
|
|
|
return mColumns.at(column);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::addColumn(Column<ESXRecordT>* column)
|
|
{
|
|
mColumns.push_back(column);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::merge()
|
|
{
|
|
for (typename std::vector<std::unique_ptr<Record<ESXRecordT>>>::iterator iter(mRecords.begin());
|
|
iter != mRecords.end(); ++iter)
|
|
(*iter)->merge();
|
|
|
|
purge();
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::purge()
|
|
{
|
|
int i = 0;
|
|
|
|
while (i < static_cast<int>(mRecords.size()))
|
|
{
|
|
if (mRecords[i]->isErased())
|
|
removeRows(i, 1);
|
|
else
|
|
++i;
|
|
}
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::removeRows(int index, int count)
|
|
{
|
|
mRecords.erase(mRecords.begin() + index, mRecords.begin() + index + count);
|
|
|
|
auto iter = mIndex.begin();
|
|
|
|
while (iter != mIndex.end())
|
|
{
|
|
if (iter->second >= index)
|
|
{
|
|
if (iter->second >= index + count)
|
|
{
|
|
iter->second -= count;
|
|
++iter;
|
|
}
|
|
else
|
|
{
|
|
iter = mIndex.erase(iter);
|
|
}
|
|
}
|
|
else
|
|
++iter;
|
|
}
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::appendBlankRecord(const ESM::RefId& id, UniversalId::Type type)
|
|
{
|
|
ESXRecordT record;
|
|
setRecordId(id, record);
|
|
record.blank();
|
|
|
|
if constexpr (std::is_same_v<ESXRecordT, ESM::Dialogue>)
|
|
{
|
|
record.mStringId = record.mId.getRefIdString();
|
|
}
|
|
|
|
auto record2 = std::make_unique<Record<ESXRecordT>>();
|
|
record2->mState = Record<ESXRecordT>::State_ModifiedOnly;
|
|
record2->mModified = std::move(record);
|
|
|
|
insertRecord(std::move(record2), getAppendIndex(id, type), type);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::searchId(const ESM::RefId& id) const
|
|
{
|
|
const auto iter = mIndex.find(id);
|
|
|
|
if (iter == mIndex.end())
|
|
return -1;
|
|
|
|
return iter->second;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::replace(int index, std::unique_ptr<RecordBase> record)
|
|
{
|
|
std::unique_ptr<Record<ESXRecordT>> tmp(static_cast<Record<ESXRecordT>*>(record.release()));
|
|
mRecords.at(index) = std::move(tmp);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::appendRecord(std::unique_ptr<RecordBase> record, UniversalId::Type type)
|
|
{
|
|
int index = getAppendIndex(getRecordId(static_cast<Record<ESXRecordT>*>(record.get())->get()), type);
|
|
insertRecord(std::move(record), index, type);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
int Collection<ESXRecordT>::getAppendIndex(const ESM::RefId& id, UniversalId::Type type) const
|
|
{
|
|
return static_cast<int>(mRecords.size());
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
std::vector<ESM::RefId> Collection<ESXRecordT>::getIds(bool listDeleted) const
|
|
{
|
|
std::vector<ESM::RefId> ids;
|
|
|
|
for (auto iter = mIndex.begin(); iter != mIndex.end(); ++iter)
|
|
{
|
|
if (listDeleted || !mRecords[iter->second]->isDeleted())
|
|
ids.push_back(getRecordId(mRecords[iter->second]->get()));
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
const Record<ESXRecordT>& Collection<ESXRecordT>::getRecord(const ESM::RefId& id) const
|
|
{
|
|
int index = getIndex(id);
|
|
return *mRecords.at(index);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
const Record<ESXRecordT>& Collection<ESXRecordT>::getRecord(int index) const
|
|
{
|
|
return *mRecords.at(index);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::insertRecord(std::unique_ptr<RecordBase> record, int index, UniversalId::Type type)
|
|
{
|
|
int size = static_cast<int>(mRecords.size());
|
|
if (index < 0 || index > size)
|
|
throw std::runtime_error("index out of range");
|
|
|
|
std::unique_ptr<Record<ESXRecordT>> record2(static_cast<Record<ESXRecordT>*>(record.release()));
|
|
ESM::RefId id = getRecordId(record2->get());
|
|
|
|
if (index == size)
|
|
mRecords.push_back(std::move(record2));
|
|
else
|
|
mRecords.insert(mRecords.begin() + index, std::move(record2));
|
|
|
|
if (index < size - 1)
|
|
{
|
|
for (auto& [key, value] : mIndex)
|
|
{
|
|
if (value >= index)
|
|
++value;
|
|
}
|
|
}
|
|
|
|
mIndex.insert(std::make_pair(id, index));
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
void Collection<ESXRecordT>::setRecord(int index, std::unique_ptr<Record<ESXRecordT>> record)
|
|
{
|
|
if (getRecordId(mRecords.at(index)->get()) != getRecordId(record->get()))
|
|
throw std::runtime_error("attempt to change the ID of a record");
|
|
|
|
mRecords.at(index) = std::move(record);
|
|
}
|
|
|
|
template <typename ESXRecordT>
|
|
bool Collection<ESXRecordT>::reorderRows(int baseIndex, const std::vector<int>& newOrder)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
#endif
|