2012-11-26 11:29:22 +00:00
|
|
|
#include "idtable.hpp"
|
2014-06-19 16:46:09 +00:00
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <QModelIndex>
|
|
|
|
#include <QVariant>
|
|
|
|
|
2017-09-30 04:04:52 +00:00
|
|
|
#include <algorithm>
|
2017-09-05 23:29:07 +00:00
|
|
|
#include <cstdint>
|
2017-09-04 23:31:09 +00:00
|
|
|
#include <limits>
|
2017-09-23 02:43:45 +00:00
|
|
|
#include <map>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <stddef.h>
|
2015-04-09 09:11:19 +00:00
|
|
|
#include <stdexcept>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <type_traits>
|
|
|
|
|
|
|
|
#include <apps/opencs/model/world/columns.hpp>
|
|
|
|
#include <apps/opencs/model/world/idtablebase.hpp>
|
|
|
|
#include <apps/opencs/model/world/record.hpp>
|
|
|
|
#include <apps/opencs/model/world/universalid.hpp>
|
2015-04-09 09:11:19 +00:00
|
|
|
|
2022-01-22 14:58:41 +00:00
|
|
|
#include <components/esm3/cellid.hpp>
|
2022-08-02 22:00:54 +00:00
|
|
|
#include <components/misc/strings/lower.hpp>
|
2016-07-02 17:48:11 +00:00
|
|
|
|
2013-06-17 09:42:02 +00:00
|
|
|
#include "collectionbase.hpp"
|
|
|
|
#include "columnbase.hpp"
|
2017-09-04 23:31:09 +00:00
|
|
|
#include "landtexture.hpp"
|
2012-11-26 11:29:22 +00:00
|
|
|
|
2014-06-05 08:28:10 +00:00
|
|
|
CSMWorld::IdTable::IdTable(CollectionBase* idCollection, unsigned int features)
|
2014-07-04 11:24:35 +00:00
|
|
|
: IdTableBase(features)
|
|
|
|
, mIdCollection(idCollection)
|
2013-11-14 10:39:14 +00:00
|
|
|
{
|
|
|
|
}
|
2012-11-26 11:29:22 +00:00
|
|
|
|
|
|
|
CSMWorld::IdTable::~IdTable() {}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::rowCount(const QModelIndex& parent) const
|
|
|
|
{
|
2015-04-02 09:19:15 +00:00
|
|
|
if (parent.isValid())
|
|
|
|
return 0;
|
2012-11-26 11:29:22 +00:00
|
|
|
|
|
|
|
return mIdCollection->getSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::columnCount(const QModelIndex& parent) const
|
|
|
|
{
|
2015-04-02 09:19:15 +00:00
|
|
|
if (parent.isValid())
|
|
|
|
return 0;
|
2015-03-06 03:36:13 +00:00
|
|
|
|
2012-11-29 17:56:28 +00:00
|
|
|
return mIdCollection->getColumns();
|
2012-11-26 11:29:22 +00:00
|
|
|
}
|
|
|
|
|
2015-01-16 10:27:17 +00:00
|
|
|
QVariant CSMWorld::IdTable::data(const QModelIndex& index, int role) const
|
2012-11-26 11:29:22 +00:00
|
|
|
{
|
2015-01-16 10:27:17 +00:00
|
|
|
if (index.row() < 0 || index.column() < 0)
|
|
|
|
return QVariant();
|
|
|
|
|
2015-06-02 21:00:39 +00:00
|
|
|
if (role == ColumnBase::Role_Display)
|
|
|
|
return QVariant(mIdCollection->getColumn(index.column()).mDisplayType);
|
|
|
|
|
2015-01-16 10:27:17 +00:00
|
|
|
if (role == ColumnBase::Role_ColumnId)
|
|
|
|
return QVariant(getColumnId(index.column()));
|
|
|
|
|
|
|
|
if ((role != Qt::DisplayRole && role != Qt::EditRole))
|
2012-11-26 11:29:22 +00:00
|
|
|
return QVariant();
|
|
|
|
|
2012-12-13 13:53:16 +00:00
|
|
|
if (role == Qt::EditRole && !mIdCollection->getColumn(index.column()).isEditable())
|
2012-12-11 14:35:47 +00:00
|
|
|
return QVariant();
|
2012-11-26 11:29:22 +00:00
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
return mIdCollection->getData(index.row(), index.column());
|
2012-11-26 11:29:22 +00:00
|
|
|
}
|
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
QVariant CSMWorld::IdTable::headerData(int section, Qt::Orientation orientation, int role) const
|
2012-11-26 11:29:22 +00:00
|
|
|
{
|
|
|
|
if (orientation == Qt::Vertical)
|
|
|
|
return QVariant();
|
|
|
|
|
2015-04-09 09:11:19 +00:00
|
|
|
if (orientation != Qt::Horizontal)
|
|
|
|
throw std::logic_error("Unknown header orientation specified");
|
|
|
|
|
2012-12-13 13:53:16 +00:00
|
|
|
if (role == Qt::DisplayRole)
|
2013-08-07 07:36:05 +00:00
|
|
|
return tr(mIdCollection->getColumn(section).getTitle().c_str());
|
2012-12-13 13:53:16 +00:00
|
|
|
|
|
|
|
if (role == ColumnBase::Role_Flags)
|
|
|
|
return mIdCollection->getColumn(section).mFlags;
|
|
|
|
|
2013-01-03 10:20:25 +00:00
|
|
|
if (role == ColumnBase::Role_Display)
|
|
|
|
return mIdCollection->getColumn(section).mDisplayType;
|
2014-06-19 16:46:09 +00:00
|
|
|
|
2015-01-16 10:27:17 +00:00
|
|
|
if (role == ColumnBase::Role_ColumnId)
|
|
|
|
return getColumnId(section);
|
|
|
|
|
2014-06-19 16:46:09 +00:00
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2014-06-17 16:28:49 +00:00
|
|
|
bool CSMWorld::IdTable::setData(const QModelIndex& index, const QVariant& value, int role)
|
2012-11-29 13:45:34 +00:00
|
|
|
{
|
2014-07-07 10:07:29 +00:00
|
|
|
if (mIdCollection->getColumn(index.column()).isEditable() && role == Qt::EditRole)
|
|
|
|
{
|
|
|
|
mIdCollection->setData(index.row(), index.column(), value);
|
2015-03-06 03:36:13 +00:00
|
|
|
|
2015-08-22 10:38:38 +00:00
|
|
|
int stateColumn = searchColumnIndex(Columns::ColumnId_Modification);
|
2018-07-05 16:03:55 +00:00
|
|
|
if (stateColumn != -1)
|
2015-08-22 10:38:38 +00:00
|
|
|
{
|
2018-07-05 16:03:55 +00:00
|
|
|
if (index.column() == stateColumn)
|
|
|
|
{
|
|
|
|
// modifying the state column can modify other values. we need to tell
|
|
|
|
// views that the whole row has changed.
|
|
|
|
|
2018-07-09 14:05:06 +00:00
|
|
|
emit dataChanged(
|
2020-05-06 15:39:32 +00:00
|
|
|
this->index(index.row(), 0), this->index(index.row(), columnCount(index.parent()) - 1));
|
2018-07-05 16:03:55 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-09 14:05:06 +00:00
|
|
|
emit dataChanged(index, index);
|
|
|
|
|
2018-07-05 16:10:43 +00:00
|
|
|
// Modifying a value can also change the Modified status of a record.
|
2018-07-05 16:03:55 +00:00
|
|
|
QModelIndex stateIndex = this->index(index.row(), stateColumn);
|
|
|
|
emit dataChanged(stateIndex, stateIndex);
|
|
|
|
}
|
2018-07-09 14:05:06 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
emit dataChanged(index, index);
|
2015-03-06 03:36:13 +00:00
|
|
|
|
2014-07-07 10:07:29 +00:00
|
|
|
return true;
|
2015-03-06 03:36:13 +00:00
|
|
|
}
|
|
|
|
|
2012-11-29 13:45:34 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::ItemFlags CSMWorld::IdTable::flags(const QModelIndex& index) const
|
|
|
|
{
|
2015-05-30 19:01:21 +00:00
|
|
|
if (!index.isValid())
|
2020-06-22 06:05:25 +00:00
|
|
|
return Qt::ItemFlags();
|
2015-05-30 19:01:21 +00:00
|
|
|
|
2012-11-29 13:45:34 +00:00
|
|
|
Qt::ItemFlags flags = Qt::ItemIsSelectable | Qt::ItemIsEnabled;
|
|
|
|
|
2012-12-13 13:53:16 +00:00
|
|
|
if (mIdCollection->getColumn(index.column()).isUserEditable())
|
2012-11-29 13:45:34 +00:00
|
|
|
flags |= Qt::ItemIsEditable;
|
|
|
|
|
2021-08-29 05:27:59 +00:00
|
|
|
int blockedColumn = searchColumnIndex(Columns::ColumnId_Blocked);
|
|
|
|
if (blockedColumn != -1 && blockedColumn != index.column())
|
|
|
|
{
|
|
|
|
bool isBlocked = mIdCollection->getData(index.row(), blockedColumn).toInt();
|
|
|
|
if (isBlocked)
|
|
|
|
flags = Qt::ItemIsSelectable; // not enabled (to grey out)
|
|
|
|
}
|
|
|
|
|
2012-11-29 13:45:34 +00:00
|
|
|
return flags;
|
2012-12-03 20:44:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CSMWorld::IdTable::removeRows(int row, int count, const QModelIndex& parent)
|
|
|
|
{
|
|
|
|
if (parent.isValid())
|
2015-04-02 09:19:15 +00:00
|
|
|
return false;
|
2012-12-03 20:44:16 +00:00
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
beginRemoveRows(parent, row, row + count - 1);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
mIdCollection->removeRows(row, count);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
|
|
|
endRemoveRows();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-03-21 09:07:25 +00:00
|
|
|
QModelIndex CSMWorld::IdTable::index(int row, int column, const QModelIndex& parent) const
|
|
|
|
{
|
|
|
|
if (parent.isValid())
|
2015-04-02 09:19:15 +00:00
|
|
|
return QModelIndex();
|
2013-03-21 09:07:25 +00:00
|
|
|
|
|
|
|
if (row < 0 || row >= mIdCollection->getSize())
|
|
|
|
return QModelIndex();
|
|
|
|
|
|
|
|
if (column < 0 || column >= mIdCollection->getColumns())
|
|
|
|
return QModelIndex();
|
|
|
|
|
2015-04-02 09:19:15 +00:00
|
|
|
return createIndex(row, column);
|
2013-03-21 09:07:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex CSMWorld::IdTable::parent(const QModelIndex& index) const
|
|
|
|
{
|
2015-04-02 09:19:15 +00:00
|
|
|
return QModelIndex();
|
2013-03-21 09:07:25 +00:00
|
|
|
}
|
|
|
|
|
2013-07-30 08:27:17 +00:00
|
|
|
void CSMWorld::IdTable::addRecord(const std::string& id, UniversalId::Type type)
|
2012-12-03 20:44:16 +00:00
|
|
|
{
|
2013-10-22 09:21:12 +00:00
|
|
|
int index = mIdCollection->getAppendIndex(id, type);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
|
|
|
beginInsertRows(QModelIndex(), index, index);
|
|
|
|
|
2013-07-30 08:27:17 +00:00
|
|
|
mIdCollection->appendBlankRecord(id, type);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
2016-03-01 10:21:06 +00:00
|
|
|
void CSMWorld::IdTable::addRecordWithData(
|
|
|
|
const std::string& id, const std::map<int, QVariant>& data, UniversalId::Type type)
|
|
|
|
{
|
|
|
|
int index = mIdCollection->getAppendIndex(id, type);
|
|
|
|
|
|
|
|
beginInsertRows(QModelIndex(), index, index);
|
|
|
|
|
|
|
|
mIdCollection->appendBlankRecord(id, type);
|
|
|
|
|
|
|
|
for (std::map<int, QVariant>::const_iterator iter(data.begin()); iter != data.end(); ++iter)
|
2016-05-06 12:04:15 +00:00
|
|
|
{
|
2016-05-08 10:05:52 +00:00
|
|
|
mIdCollection->setData(index, iter->first, iter->second);
|
2016-05-06 12:04:15 +00:00
|
|
|
}
|
2016-03-01 10:21:06 +00:00
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
2014-01-20 12:59:00 +00:00
|
|
|
void CSMWorld::IdTable::cloneRecord(
|
2014-01-19 15:49:39 +00:00
|
|
|
const std::string& origin, const std::string& destination, CSMWorld::UniversalId::Type type)
|
|
|
|
{
|
2021-07-24 20:00:25 +00:00
|
|
|
int index = mIdCollection->getAppendIndex(destination, type);
|
2014-05-25 13:46:23 +00:00
|
|
|
|
2014-01-20 17:28:06 +00:00
|
|
|
beginInsertRows(QModelIndex(), index, index);
|
2014-01-27 12:08:14 +00:00
|
|
|
mIdCollection->cloneRecord(origin, destination, type);
|
2014-01-20 17:28:06 +00:00
|
|
|
endInsertRows();
|
2014-01-19 15:49:39 +00:00
|
|
|
}
|
|
|
|
|
2017-09-01 02:01:38 +00:00
|
|
|
bool CSMWorld::IdTable::touchRecord(const std::string& id)
|
|
|
|
{
|
|
|
|
bool changed = mIdCollection->touchRecord(id);
|
|
|
|
|
|
|
|
int row = mIdCollection->getIndex(id);
|
|
|
|
int column = mIdCollection->searchColumnIndex(Columns::ColumnId_RecordType);
|
|
|
|
if (changed && column != -1)
|
|
|
|
{
|
|
|
|
QModelIndex modelIndex = index(row, column);
|
|
|
|
emit dataChanged(modelIndex, modelIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string CSMWorld::IdTable::getId(int row) const
|
|
|
|
{
|
|
|
|
return mIdCollection->getId(row);
|
|
|
|
}
|
|
|
|
|
2014-06-02 18:41:37 +00:00
|
|
|
/// This method can return only indexes to the top level table cells
|
2012-12-03 20:44:16 +00:00
|
|
|
QModelIndex CSMWorld::IdTable::getModelIndex(const std::string& id, int column) const
|
|
|
|
{
|
2018-12-23 16:05:00 +00:00
|
|
|
int row = mIdCollection->searchId(id);
|
|
|
|
if (row != -1)
|
|
|
|
return index(row, column);
|
|
|
|
|
|
|
|
return QModelIndex();
|
2012-12-06 13:56:04 +00:00
|
|
|
}
|
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
void CSMWorld::IdTable::setRecord(
|
|
|
|
const std::string& id, std::unique_ptr<RecordBase> record, CSMWorld::UniversalId::Type type)
|
2012-12-06 13:56:04 +00:00
|
|
|
{
|
2013-04-29 15:52:01 +00:00
|
|
|
int index = mIdCollection->searchId(id);
|
2012-12-06 13:56:04 +00:00
|
|
|
|
|
|
|
if (index == -1)
|
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
// For info records, appendRecord may use a different index than the one returned by
|
|
|
|
// getAppendIndex (because of prev/next links). This can result in the display not
|
|
|
|
// updating correctly after an undo
|
|
|
|
//
|
|
|
|
// Use an alternative method to get the correct index. For non-Info records the
|
|
|
|
// record pointer is ignored and internally calls getAppendIndex.
|
|
|
|
int index2 = mIdCollection->getInsertIndex(id, type, record.get());
|
2012-12-06 13:56:04 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
beginInsertRows(QModelIndex(), index2, index2);
|
2012-12-06 13:56:04 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
mIdCollection->appendRecord(std::move(record), type);
|
2012-12-06 13:56:04 +00:00
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-07-23 04:21:21 +00:00
|
|
|
mIdCollection->replace(index, std::move(record));
|
2012-12-06 13:56:04 +00:00
|
|
|
emit dataChanged(
|
|
|
|
CSMWorld::IdTable::index(index, 0), CSMWorld::IdTable::index(index, mIdCollection->getColumns() - 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const CSMWorld::RecordBase& CSMWorld::IdTable::getRecord(const std::string& id) const
|
|
|
|
{
|
|
|
|
return mIdCollection->getRecord(id);
|
2013-08-08 10:49:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::searchColumnIndex(Columns::ColumnId id) const
|
|
|
|
{
|
2013-09-27 13:04:09 +00:00
|
|
|
return mIdCollection->searchColumnIndex(id);
|
2013-08-08 10:49:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::findColumnIndex(Columns::ColumnId id) const
|
|
|
|
{
|
2013-09-27 13:04:09 +00:00
|
|
|
return mIdCollection->findColumnIndex(id);
|
2013-11-14 10:39:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSMWorld::IdTable::reorderRows(int baseIndex, const std::vector<int>& newOrder)
|
|
|
|
{
|
|
|
|
if (!newOrder.empty())
|
|
|
|
if (mIdCollection->reorderRows(baseIndex, newOrder))
|
|
|
|
emit dataChanged(index(baseIndex, 0),
|
2021-05-02 06:43:44 +00:00
|
|
|
index(baseIndex + static_cast<int>(newOrder.size()) - 1, mIdCollection->getColumns() - 1));
|
2013-11-14 10:39:14 +00:00
|
|
|
}
|
|
|
|
|
2014-03-02 21:43:44 +00:00
|
|
|
std::pair<CSMWorld::UniversalId, std::string> CSMWorld::IdTable::view(int row) const
|
|
|
|
{
|
|
|
|
std::string id;
|
|
|
|
std::string hint;
|
|
|
|
|
2014-07-04 11:24:35 +00:00
|
|
|
if (getFeatures() & Feature_ViewCell)
|
2014-03-02 21:43:44 +00:00
|
|
|
{
|
|
|
|
int cellColumn = mIdCollection->searchColumnIndex(Columns::ColumnId_Cell);
|
|
|
|
int idColumn = mIdCollection->searchColumnIndex(Columns::ColumnId_Id);
|
|
|
|
|
|
|
|
if (cellColumn != -1 && idColumn != -1)
|
|
|
|
{
|
|
|
|
id = mIdCollection->getData(row, cellColumn).toString().toUtf8().constData();
|
|
|
|
hint = "r:" + std::string(mIdCollection->getData(row, idColumn).toString().toUtf8().constData());
|
|
|
|
}
|
|
|
|
}
|
2014-07-04 11:24:35 +00:00
|
|
|
else if (getFeatures() & Feature_ViewId)
|
2014-03-02 21:43:44 +00:00
|
|
|
{
|
|
|
|
int column = mIdCollection->searchColumnIndex(Columns::ColumnId_Id);
|
|
|
|
|
|
|
|
if (column != -1)
|
|
|
|
{
|
|
|
|
id = mIdCollection->getData(row, column).toString().toUtf8().constData();
|
|
|
|
hint = "c:" + id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (id.empty())
|
|
|
|
return std::make_pair(UniversalId::Type_None, "");
|
|
|
|
|
|
|
|
if (id[0] == '#')
|
2016-07-02 17:48:11 +00:00
|
|
|
id = ESM::CellId::sDefaultWorldspace;
|
2014-03-02 21:43:44 +00:00
|
|
|
|
|
|
|
return std::make_pair(UniversalId(UniversalId::Type_Scene, id), hint);
|
2014-03-08 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
2014-06-02 18:41:37 +00:00
|
|
|
/// For top level data/columns
|
2014-07-04 11:24:35 +00:00
|
|
|
bool CSMWorld::IdTable::isDeleted(const std::string& id) const
|
|
|
|
{
|
|
|
|
return getRecord(id).isDeleted();
|
|
|
|
}
|
|
|
|
|
2014-03-08 14:27:43 +00:00
|
|
|
int CSMWorld::IdTable::getColumnId(int column) const
|
|
|
|
{
|
|
|
|
return mIdCollection->getColumn(column).getId();
|
2014-06-09 08:35:39 +00:00
|
|
|
}
|
2015-04-09 09:11:19 +00:00
|
|
|
|
|
|
|
CSMWorld::CollectionBase* CSMWorld::IdTable::idCollection() const
|
|
|
|
{
|
|
|
|
return mIdCollection;
|
|
|
|
}
|
2017-08-25 23:51:30 +00:00
|
|
|
|
|
|
|
CSMWorld::LandTextureIdTable::LandTextureIdTable(CollectionBase* idCollection, unsigned int features)
|
|
|
|
: IdTable(idCollection, features)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-09-04 23:31:09 +00:00
|
|
|
CSMWorld::LandTextureIdTable::ImportResults CSMWorld::LandTextureIdTable::importTextures(
|
|
|
|
const std::vector<std::string>& ids)
|
|
|
|
{
|
|
|
|
ImportResults results;
|
|
|
|
|
2017-09-23 02:43:45 +00:00
|
|
|
// Map existing textures to ids
|
|
|
|
std::map<std::string, std::string> reverseLookupMap;
|
|
|
|
for (int i = 0; i < idCollection()->getSize(); ++i)
|
|
|
|
{
|
|
|
|
auto& record = static_cast<const Record<LandTexture>&>(idCollection()->getRecord(i));
|
|
|
|
if (record.isModified())
|
2022-08-02 22:01:41 +00:00
|
|
|
reverseLookupMap.emplace(Misc::StringUtils::lowerCase(record.get().mTexture), idCollection()->getId(i));
|
2017-09-23 02:43:45 +00:00
|
|
|
}
|
|
|
|
|
2017-09-04 23:31:09 +00:00
|
|
|
for (const std::string& id : ids)
|
|
|
|
{
|
|
|
|
int plugin, index;
|
|
|
|
|
|
|
|
LandTexture::parseUniqueRecordId(id, plugin, index);
|
|
|
|
int oldRow = idCollection()->searchId(id);
|
|
|
|
|
|
|
|
// If it does not exist or it is in the current plugin, it can be skipped.
|
2017-10-14 17:18:54 +00:00
|
|
|
if (oldRow < 0 || plugin == 0)
|
2017-09-04 23:31:09 +00:00
|
|
|
{
|
2020-10-17 08:26:35 +00:00
|
|
|
results.recordMapping.emplace_back(id, id);
|
2017-09-04 23:31:09 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-09-23 02:43:45 +00:00
|
|
|
// Look for a pre-existing record
|
|
|
|
auto& record = static_cast<const Record<LandTexture>&>(idCollection()->getRecord(oldRow));
|
2021-11-10 20:25:16 +00:00
|
|
|
std::string texture = Misc::StringUtils::lowerCase(record.get().mTexture);
|
2017-09-30 04:04:52 +00:00
|
|
|
auto searchIt = reverseLookupMap.find(texture);
|
2017-09-23 02:43:45 +00:00
|
|
|
if (searchIt != reverseLookupMap.end())
|
|
|
|
{
|
2020-10-17 08:26:35 +00:00
|
|
|
results.recordMapping.emplace_back(id, searchIt->second);
|
2017-09-23 02:43:45 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iterate until an unused index or found, or the index has completely wrapped around.
|
2017-09-09 19:37:52 +00:00
|
|
|
int startIndex = index;
|
2017-09-04 23:31:09 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
std::string newId = LandTexture::createUniqueRecordId(0, index);
|
|
|
|
int newRow = idCollection()->searchId(newId);
|
|
|
|
|
|
|
|
if (newRow < 0)
|
|
|
|
{
|
|
|
|
// Id not taken, clone it
|
|
|
|
cloneRecord(id, newId, UniversalId::Type_LandTexture);
|
|
|
|
results.createdRecords.push_back(newId);
|
2020-10-17 08:26:35 +00:00
|
|
|
results.recordMapping.emplace_back(id, newId);
|
2017-09-30 04:04:52 +00:00
|
|
|
reverseLookupMap.emplace(texture, newId);
|
2017-09-04 23:31:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-09-23 02:43:45 +00:00
|
|
|
const size_t MaxIndex = std::numeric_limits<uint16_t>::max() - 1;
|
|
|
|
index = (index + 1) % MaxIndex;
|
2017-09-09 19:37:52 +00:00
|
|
|
} while (index != startIndex);
|
2017-09-04 23:31:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return results;
|
|
|
|
}
|