1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 15:23:58 +00:00
openmw/apps/opencs/model/world/commands.cpp

568 lines
16 KiB
C++
Raw Normal View History

2012-11-29 15:05:28 +00:00
#include "commands.hpp"
2014-01-27 13:43:12 +00:00
2022-10-19 17:02:00 +00:00
#include <algorithm>
#include <cmath>
#include <sstream>
2017-09-04 23:31:09 +00:00
#include <unordered_set>
2022-10-19 17:02:00 +00:00
#include <apps/opencs/model/world/columns.hpp>
#include <apps/opencs/model/world/land.hpp>
#include <apps/opencs/model/world/landtexture.hpp>
#include <apps/opencs/model/world/record.hpp>
#include <apps/opencs/model/world/universalid.hpp>
#include <components/esm3/loadland.hpp>
#include <components/esm3/loadpgrd.hpp>
#include <components/misc/constants.hpp>
2014-01-27 13:43:12 +00:00
#include <QAbstractItemModel>
#include <QAbstractProxyModel>
2014-01-27 13:45:07 +00:00
#include "cellcoordinates.hpp"
2014-01-27 13:45:07 +00:00
#include "idtable.hpp"
#include "idtree.hpp"
#include "nestedtablewrapper.hpp"
2016-05-19 22:33:15 +00:00
#include "pathgrid.hpp"
CSMWorld::TouchCommand::TouchCommand(IdTable& table, const std::string& id, QUndoCommand* parent)
: QUndoCommand(parent)
, mTable(table)
, mId(id)
, mOld(nullptr)
, mChanged(false)
{
setText(("Touch " + mId).c_str());
}
void CSMWorld::TouchCommand::redo()
{
mOld.reset(mTable.getRecord(mId).clone().get());
mChanged = mTable.touchRecord(mId);
}
void CSMWorld::TouchCommand::undo()
{
if (mChanged)
{
mTable.setRecord(mId, std::move(mOld));
mChanged = false;
}
}
2022-09-22 18:26:05 +00:00
CSMWorld::ImportLandTexturesCommand::ImportLandTexturesCommand(
IdTable& landTable, IdTable& ltexTable, QUndoCommand* parent)
2017-09-04 23:31:09 +00:00
: QUndoCommand(parent)
, mLands(landTable)
, mLtexs(ltexTable)
, mOldState(0)
2017-09-04 23:31:09 +00:00
{
setText("Copy land textures to current plugin");
2017-09-04 23:31:09 +00:00
}
void CSMWorld::ImportLandTexturesCommand::redo()
2017-09-04 23:31:09 +00:00
{
int pluginColumn = mLands.findColumnIndex(Columns::ColumnId_PluginIndex);
int oldPlugin = mLands.data(mLands.getModelIndex(getOriginId(), pluginColumn)).toInt();
2017-09-04 23:31:09 +00:00
// Original data
int textureColumn = mLands.findColumnIndex(Columns::ColumnId_LandTexturesIndex);
2017-09-23 01:29:40 +00:00
mOld = mLands.data(mLands.getModelIndex(getOriginId(), textureColumn)).value<DataType>();
// Need to make a copy so the old values can be looked up
2017-09-23 01:29:40 +00:00
DataType copy(mOld);
// Perform touch/copy/etc...
onRedo();
// Find all indices used
std::unordered_set<int> texIndices;
2017-09-23 01:29:40 +00:00
for (int i = 0; i < mOld.size(); ++i)
2017-09-04 23:31:09 +00:00
{
// All indices are offset by 1 for a default texture
2017-09-23 01:29:40 +00:00
if (mOld[i] > 0)
texIndices.insert(mOld[i] - 1);
}
2017-09-04 23:31:09 +00:00
std::vector<std::string> oldTextures;
oldTextures.reserve(texIndices.size());
for (int index : texIndices)
{
oldTextures.push_back(LandTexture::createUniqueRecordId(oldPlugin, index));
}
2017-09-04 23:31:09 +00:00
// Import the textures, replace old values
LandTextureIdTable::ImportResults results = dynamic_cast<LandTextureIdTable&>(mLtexs).importTextures(oldTextures);
mCreatedTextures = std::move(results.createdRecords);
for (const auto& it : results.recordMapping)
{
int plugin = 0, newIndex = 0, oldIndex = 0;
LandTexture::parseUniqueRecordId(it.first, plugin, oldIndex);
LandTexture::parseUniqueRecordId(it.second, plugin, newIndex);
2017-09-04 23:31:09 +00:00
if (newIndex != oldIndex)
{
for (int i = 0; i < Land::LAND_NUM_TEXTURES; ++i)
2017-09-04 23:31:09 +00:00
{
// All indices are offset by 1 for a default texture
2017-09-23 01:29:40 +00:00
if (mOld[i] == oldIndex + 1)
copy[i] = newIndex + 1;
2017-09-04 23:31:09 +00:00
}
}
}
2017-09-04 23:31:09 +00:00
// Apply modification
int stateColumn = mLands.findColumnIndex(Columns::ColumnId_Modification);
mOldState = mLands.data(mLands.getModelIndex(getDestinationId(), stateColumn)).toInt();
2017-09-23 01:29:40 +00:00
QVariant variant;
variant.setValue(copy);
mLands.setData(mLands.getModelIndex(getDestinationId(), textureColumn), variant);
}
void CSMWorld::ImportLandTexturesCommand::undo()
{
// Restore to previous
int textureColumn = mLands.findColumnIndex(Columns::ColumnId_LandTexturesIndex);
2017-09-23 01:29:40 +00:00
QVariant variant;
variant.setValue(mOld);
mLands.setData(mLands.getModelIndex(getDestinationId(), textureColumn), variant);
int stateColumn = mLands.findColumnIndex(Columns::ColumnId_Modification);
mLands.setData(mLands.getModelIndex(getDestinationId(), stateColumn), mOldState);
// Undo copy/touch/etc...
onUndo();
for (const std::string& id : mCreatedTextures)
{
int row = mLtexs.getModelIndex(id, 0).row();
mLtexs.removeRows(row, 1);
2017-09-04 23:31:09 +00:00
}
mCreatedTextures.clear();
2017-09-04 23:31:09 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::CopyLandTexturesCommand::CopyLandTexturesCommand(
IdTable& landTable, IdTable& ltexTable, const std::string& origin, const std::string& dest, QUndoCommand* parent)
: ImportLandTexturesCommand(landTable, ltexTable, parent)
, mOriginId(origin)
, mDestId(dest)
{
}
const std::string& CSMWorld::CopyLandTexturesCommand::getOriginId() const
{
return mOriginId;
}
const std::string& CSMWorld::CopyLandTexturesCommand::getDestinationId() const
{
return mDestId;
}
2022-09-22 18:26:05 +00:00
CSMWorld::TouchLandCommand::TouchLandCommand(
IdTable& landTable, IdTable& ltexTable, const std::string& id, QUndoCommand* parent)
: ImportLandTexturesCommand(landTable, ltexTable, parent)
, mId(id)
, mOld(nullptr)
, mChanged(false)
{
setText(("Touch " + mId).c_str());
}
const std::string& CSMWorld::TouchLandCommand::getOriginId() const
{
return mId;
}
const std::string& CSMWorld::TouchLandCommand::getDestinationId() const
{
return mId;
}
void CSMWorld::TouchLandCommand::onRedo()
{
mChanged = mLands.touchRecord(mId);
2022-09-22 18:26:05 +00:00
if (mChanged)
mOld.reset(mLands.getRecord(mId).clone().get());
}
void CSMWorld::TouchLandCommand::onUndo()
2017-09-04 23:31:09 +00:00
{
if (mChanged)
{
mLands.setRecord(mId, std::move(mOld));
2017-09-04 23:31:09 +00:00
mChanged = false;
}
}
2022-09-22 18:26:05 +00:00
CSMWorld::ModifyCommand::ModifyCommand(
QAbstractItemModel& model, const QModelIndex& index, const QVariant& new_, QUndoCommand* parent)
: QUndoCommand(parent)
, mModel(&model)
, mIndex(index)
, mNew(new_)
, mHasRecordState(false)
, mOldRecordState(CSMWorld::RecordBase::State_BaseOnly)
{
2022-09-22 18:26:05 +00:00
if (QAbstractProxyModel* proxy = dynamic_cast<QAbstractProxyModel*>(mModel))
{
// Replace proxy with actual model
2022-09-22 18:26:05 +00:00
mIndex = proxy->mapToSource(mIndex);
mModel = proxy->sourceModel();
}
2021-08-21 10:42:55 +00:00
}
2021-08-21 10:42:55 +00:00
void CSMWorld::ModifyCommand::redo()
{
if (mIndex.parent().isValid())
{
2018-11-15 13:50:23 +00:00
CSMWorld::IdTree* tree = &dynamic_cast<CSMWorld::IdTree&>(*mModel);
2022-09-22 18:26:05 +00:00
setText("Modify "
+ tree->nestedHeaderData(mIndex.parent().column(), mIndex.column(), Qt::Horizontal, Qt::DisplayRole)
.toString());
}
else
{
2022-09-22 18:26:05 +00:00
setText("Modify " + mModel->headerData(mIndex.column(), Qt::Horizontal, Qt::DisplayRole).toString());
}
// Remember record state before the modification
2022-09-22 18:26:05 +00:00
if (CSMWorld::IdTable* table = dynamic_cast<IdTable*>(mModel))
{
mHasRecordState = true;
int stateColumnIndex = table->findColumnIndex(Columns::ColumnId_Modification);
int rowIndex = mIndex.row();
if (mIndex.parent().isValid())
{
rowIndex = mIndex.parent().row();
2016-05-23 19:51:36 +00:00
}
mRecordStateIndex = table->index(rowIndex, stateColumnIndex);
mOldRecordState = static_cast<CSMWorld::RecordBase::State>(table->data(mRecordStateIndex).toInt());
}
2012-11-29 15:05:28 +00:00
2022-09-22 18:26:05 +00:00
mOld = mModel->data(mIndex, Qt::EditRole);
mModel->setData(mIndex, mNew);
2012-11-29 15:05:28 +00:00
}
void CSMWorld::ModifyCommand::undo()
{
2022-09-22 18:26:05 +00:00
mModel->setData(mIndex, mOld);
if (mHasRecordState)
{
mModel->setData(mRecordStateIndex, mOldRecordState);
}
}
void CSMWorld::CreateCommand::applyModifications()
{
if (!mNestedValues.empty())
{
2018-11-15 13:50:23 +00:00
CSMWorld::IdTree* tree = &dynamic_cast<CSMWorld::IdTree&>(mModel);
2022-09-22 18:26:05 +00:00
std::map<int, std::pair<int, QVariant>>::const_iterator current = mNestedValues.begin();
std::map<int, std::pair<int, QVariant>>::const_iterator end = mNestedValues.end();
for (; current != end; ++current)
{
2022-09-22 18:26:05 +00:00
QModelIndex index = tree->index(0, current->second.first, tree->getNestedModelIndex(mId, current->first));
tree->setData(index, current->second.second);
}
}
}
2022-09-22 18:26:05 +00:00
CSMWorld::CreateCommand::CreateCommand(IdTable& model, const std::string& id, QUndoCommand* parent)
: QUndoCommand(parent)
, mModel(model)
, mId(id)
, mType(UniversalId::Type_None)
{
2022-09-22 18:26:05 +00:00
setText(("Create record " + id).c_str());
}
2022-09-22 18:26:05 +00:00
void CSMWorld::CreateCommand::addValue(int column, const QVariant& value)
{
mValues[column] = value;
}
2022-09-22 18:26:05 +00:00
void CSMWorld::CreateCommand::addNestedValue(int parentColumn, int nestedColumn, const QVariant& value)
{
mNestedValues[parentColumn] = std::make_pair(nestedColumn, value);
}
2022-09-22 18:26:05 +00:00
void CSMWorld::CreateCommand::setType(UniversalId::Type type)
{
mType = type;
}
void CSMWorld::CreateCommand::redo()
{
2022-09-22 18:26:05 +00:00
mModel.addRecordWithData(mId, mValues, mType);
applyModifications();
}
void CSMWorld::CreateCommand::undo()
{
2022-09-22 18:26:05 +00:00
mModel.removeRow(mModel.getModelIndex(mId, 0).row());
2012-12-06 13:56:04 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::RevertCommand::RevertCommand(IdTable& model, const std::string& id, QUndoCommand* parent)
: QUndoCommand(parent)
, mModel(model)
, mId(id)
, mOld(nullptr)
2012-12-06 13:56:04 +00:00
{
2022-09-22 18:26:05 +00:00
setText(("Revert record " + id).c_str());
2012-12-06 13:56:04 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::RevertCommand::~RevertCommand() {}
2012-12-06 13:56:04 +00:00
void CSMWorld::RevertCommand::redo()
{
2022-09-22 18:26:05 +00:00
mOld = mModel.getRecord(mId).clone();
2022-09-22 18:26:05 +00:00
int column = mModel.findColumnIndex(Columns::ColumnId_Modification);
2013-08-26 10:25:52 +00:00
2022-09-22 18:26:05 +00:00
QModelIndex index = mModel.getModelIndex(mId, column);
RecordBase::State state = static_cast<RecordBase::State>(mModel.data(index).toInt());
2012-12-06 13:56:04 +00:00
2022-09-22 18:26:05 +00:00
if (state == RecordBase::State_ModifiedOnly)
2012-12-06 13:56:04 +00:00
{
2022-09-22 18:26:05 +00:00
mModel.removeRows(index.row(), 1);
2012-12-06 13:56:04 +00:00
}
else
{
2022-09-22 18:26:05 +00:00
mModel.setData(index, static_cast<int>(RecordBase::State_BaseOnly));
2012-12-06 13:56:04 +00:00
}
}
void CSMWorld::RevertCommand::undo()
2012-12-06 14:18:41 +00:00
{
2022-09-22 18:26:05 +00:00
mModel.setRecord(mId, std::move(mOld));
2012-12-06 14:18:41 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::DeleteCommand::DeleteCommand(
IdTable& model, const std::string& id, CSMWorld::UniversalId::Type type, QUndoCommand* parent)
: QUndoCommand(parent)
, mModel(model)
, mId(id)
, mOld(nullptr)
, mType(type)
2012-12-06 14:18:41 +00:00
{
2022-09-22 18:26:05 +00:00
setText(("Delete record " + id).c_str());
2012-12-06 14:18:41 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::DeleteCommand::~DeleteCommand() {}
2012-12-06 14:18:41 +00:00
void CSMWorld::DeleteCommand::redo()
{
2022-09-22 18:26:05 +00:00
mOld = mModel.getRecord(mId).clone();
2022-09-22 18:26:05 +00:00
int column = mModel.findColumnIndex(Columns::ColumnId_Modification);
2013-08-26 10:25:52 +00:00
2022-09-22 18:26:05 +00:00
QModelIndex index = mModel.getModelIndex(mId, column);
RecordBase::State state = static_cast<RecordBase::State>(mModel.data(index).toInt());
2012-12-06 14:18:41 +00:00
2022-09-22 18:26:05 +00:00
if (state == RecordBase::State_ModifiedOnly)
2012-12-06 14:18:41 +00:00
{
2022-09-22 18:26:05 +00:00
mModel.removeRows(index.row(), 1);
2012-12-06 14:18:41 +00:00
}
else
{
2022-09-22 18:26:05 +00:00
mModel.setData(index, static_cast<int>(RecordBase::State_Deleted));
2012-12-06 14:18:41 +00:00
}
}
void CSMWorld::DeleteCommand::undo()
2012-12-06 13:56:04 +00:00
{
2022-09-22 18:26:05 +00:00
mModel.setRecord(mId, std::move(mOld), mType);
}
2022-09-22 18:26:05 +00:00
CSMWorld::ReorderRowsCommand::ReorderRowsCommand(IdTable& model, int baseIndex, const std::vector<int>& newOrder)
: mModel(model)
, mBaseIndex(baseIndex)
, mNewOrder(newOrder)
{
}
void CSMWorld::ReorderRowsCommand::redo()
{
2022-09-22 18:26:05 +00:00
mModel.reorderRows(mBaseIndex, mNewOrder);
}
void CSMWorld::ReorderRowsCommand::undo()
{
2022-09-22 18:26:05 +00:00
int size = static_cast<int>(mNewOrder.size());
std::vector<int> reverse(size);
2022-09-22 18:26:05 +00:00
for (int i = 0; i < size; ++i)
reverse.at(mNewOrder[i]) = i;
2022-09-22 18:26:05 +00:00
mModel.reorderRows(mBaseIndex, reverse);
}
2014-01-27 13:46:58 +00:00
2022-09-22 18:26:05 +00:00
CSMWorld::CloneCommand::CloneCommand(CSMWorld::IdTable& model, const std::string& idOrigin,
const std::string& idDestination, const CSMWorld::UniversalId::Type type, QUndoCommand* parent)
: CreateCommand(model, idDestination, parent)
, mIdOrigin(idOrigin)
2014-01-27 13:46:58 +00:00
{
2022-09-22 18:26:05 +00:00
setType(type);
setText(("Clone record " + idOrigin + " to the " + idDestination).c_str());
2014-01-27 13:46:58 +00:00
}
void CSMWorld::CloneCommand::redo()
{
mModel.cloneRecord(ESM::RefId::stringRefId(mIdOrigin), ESM::RefId::stringRefId(mId), mType);
applyModifications();
for (auto& value : mOverrideValues)
{
2022-09-22 18:26:05 +00:00
mModel.setData(mModel.getModelIndex(mId, value.first), value.second);
}
2014-01-27 13:46:58 +00:00
}
void CSMWorld::CloneCommand::undo()
{
2022-09-22 18:26:05 +00:00
mModel.removeRow(mModel.getModelIndex(mId, 0).row());
}
void CSMWorld::CloneCommand::setOverrideValue(int column, QVariant value)
{
mOverrideValues.emplace_back(std::make_pair(column, value));
}
2022-09-22 18:26:05 +00:00
CSMWorld::CreatePathgridCommand::CreatePathgridCommand(IdTable& model, const std::string& id, QUndoCommand* parent)
2016-05-19 22:33:15 +00:00
: CreateCommand(model, id, parent)
{
setType(UniversalId::Type_Pathgrid);
}
void CSMWorld::CreatePathgridCommand::redo()
{
CreateCommand::redo();
2022-09-22 18:26:05 +00:00
std::unique_ptr<Record<Pathgrid>> record
= std::make_unique<Record<Pathgrid>>(static_cast<const Record<Pathgrid>&>(mModel.getRecord(mId)));
record->get().blank();
record->get().mCell = ESM::RefId::stringRefId(mId);
2016-05-19 22:33:15 +00:00
std::pair<CellCoordinates, bool> coords = CellCoordinates::fromId(mId);
if (coords.second)
2016-05-19 22:33:15 +00:00
{
record->get().mData.mX = coords.first.getX();
record->get().mData.mY = coords.first.getY();
2016-05-19 22:33:15 +00:00
}
mModel.setRecord(mId, std::move(record), mType);
2016-05-19 22:33:15 +00:00
}
2022-09-22 18:26:05 +00:00
CSMWorld::UpdateCellCommand::UpdateCellCommand(IdTable& model, int row, QUndoCommand* parent)
: QUndoCommand(parent)
, mModel(model)
, mRow(row)
{
2022-09-22 18:26:05 +00:00
setText("Update cell ID");
}
void CSMWorld::UpdateCellCommand::redo()
{
if (!mNew.isValid())
{
2022-09-22 18:26:05 +00:00
int cellColumn = mModel.searchColumnIndex(Columns::ColumnId_Cell);
mIndex = mModel.index(mRow, cellColumn);
2022-09-22 18:26:05 +00:00
QModelIndex xIndex = mModel.index(mRow, mModel.findColumnIndex(Columns::ColumnId_PositionXPos));
2022-09-22 18:26:05 +00:00
QModelIndex yIndex = mModel.index(mRow, mModel.findColumnIndex(Columns::ColumnId_PositionYPos));
2022-09-22 18:26:05 +00:00
int x = std::floor(mModel.data(xIndex).toFloat() / Constants::CellSizeInUnits);
int y = std::floor(mModel.data(yIndex).toFloat() / Constants::CellSizeInUnits);
std::ostringstream stream;
stream << "#" << x << " " << y;
2022-09-22 18:26:05 +00:00
mNew = QString::fromUtf8(stream.str().c_str());
}
2022-09-22 18:26:05 +00:00
mModel.setData(mIndex, mNew);
}
void CSMWorld::UpdateCellCommand::undo()
{
2022-09-22 18:26:05 +00:00
mModel.setData(mIndex, mOld);
}
2022-09-22 18:26:05 +00:00
CSMWorld::DeleteNestedCommand::DeleteNestedCommand(
IdTree& model, const std::string& id, int nestedRow, int parentColumn, QUndoCommand* parent)
: QUndoCommand(parent)
, NestedTableStoring(model, id, parentColumn)
, mModel(model)
, mId(id)
, mParentColumn(parentColumn)
, mNestedRow(nestedRow)
{
2022-09-22 18:26:05 +00:00
std::string title = model.headerData(parentColumn, Qt::Horizontal, Qt::DisplayRole).toString().toUtf8().constData();
setText(("Delete row in " + title + " sub-table of " + mId).c_str());
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModifyParentCommand = new ModifyCommand(mModel, parentIndex, parentIndex.data(Qt::EditRole), this);
}
void CSMWorld::DeleteNestedCommand::redo()
{
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModifyParentCommand->redo();
2022-09-22 18:26:05 +00:00
mModel.removeRows(mNestedRow, 1, parentIndex);
}
void CSMWorld::DeleteNestedCommand::undo()
{
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModel.setNestedTable(parentIndex, getOld());
mModifyParentCommand->undo();
}
2014-06-24 17:19:40 +00:00
2022-09-22 18:26:05 +00:00
CSMWorld::AddNestedCommand::AddNestedCommand(
IdTree& model, const std::string& id, int nestedRow, int parentColumn, QUndoCommand* parent)
: QUndoCommand(parent)
, NestedTableStoring(model, id, parentColumn)
, mModel(model)
, mId(id)
, mNewRow(nestedRow)
, mParentColumn(parentColumn)
2014-06-24 17:19:40 +00:00
{
2022-09-22 18:26:05 +00:00
std::string title = model.headerData(parentColumn, Qt::Horizontal, Qt::DisplayRole).toString().toUtf8().constData();
setText(("Add row in " + title + " sub-table of " + mId).c_str());
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModifyParentCommand = new ModifyCommand(mModel, parentIndex, parentIndex.data(Qt::EditRole), this);
2014-06-24 17:19:40 +00:00
}
void CSMWorld::AddNestedCommand::redo()
{
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModifyParentCommand->redo();
2022-09-22 18:26:05 +00:00
mModel.addNestedRow(parentIndex, mNewRow);
2014-06-24 17:19:40 +00:00
}
void CSMWorld::AddNestedCommand::undo()
{
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModel.setNestedTable(parentIndex, getOld());
mModifyParentCommand->undo();
}
CSMWorld::NestedTableStoring::NestedTableStoring(const IdTree& model, const std::string& id, int parentColumn)
2022-09-22 18:26:05 +00:00
: mOld(model.nestedTable(model.getModelIndex(id, parentColumn)))
{
}
CSMWorld::NestedTableStoring::~NestedTableStoring()
{
delete mOld;
2014-06-24 17:19:40 +00:00
}
const CSMWorld::NestedTableWrapperBase& CSMWorld::NestedTableStoring::getOld() const
{
return *mOld;
}