1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-20 05:53:50 +00:00
openmw-tes3mp/apps/opencs/model/world/commands.cpp

356 lines
10 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
#include <cmath>
#include <sstream>
#include <components/misc/stringops.hpp>
2014-01-27 13:43:12 +00:00
#include <QAbstractItemModel>
#include <QAbstractProxyModel>
2014-01-27 13:45:07 +00:00
#include "idtable.hpp"
#include "idtree.hpp"
#include "nestedtablewrapper.hpp"
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)
2012-11-29 15:05:28 +00:00
{
if (QAbstractProxyModel *proxy = dynamic_cast<QAbstractProxyModel *> (&model))
{
// Replace proxy with actual model
mIndex = proxy->mapToSource (index);
mModel = proxy->sourceModel();
}
if (mIndex.parent().isValid())
{
setText ("Modify " + dynamic_cast<CSMWorld::IdTree*>(mModel)->nestedHeaderData (
mIndex.parent().column(), mIndex.column(), Qt::Horizontal, Qt::DisplayRole).toString());
}
else
{
setText ("Modify " + mModel->headerData (mIndex.column(), Qt::Horizontal, Qt::DisplayRole).toString());
}
// Remember record state before the modification
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();
}
mRecordStateIndex = table->index(rowIndex, stateColumnIndex);
mOldRecordState = static_cast<CSMWorld::RecordBase::State>(table->data(mRecordStateIndex).toInt());
}
2012-11-29 15:05:28 +00:00
}
void CSMWorld::ModifyCommand::redo()
{
mOld = mModel->data (mIndex, Qt::EditRole);
mModel->setData (mIndex, mNew);
2012-11-29 15:05:28 +00:00
}
void CSMWorld::ModifyCommand::undo()
{
mModel->setData (mIndex, mOld);
if (mHasRecordState)
{
mModel->setData(mRecordStateIndex, mOldRecordState);
}
}
void CSMWorld::CreateCommand::applyModifications()
{
if (!mNestedValues.empty())
{
CSMWorld::IdTree *tree = dynamic_cast<CSMWorld::IdTree *>(&mModel);
if (tree == NULL)
{
throw std::logic_error("CSMWorld::CreateCommand: Attempt to add nested values to the non-nested model");
}
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)
{
QModelIndex index = tree->index(0,
current->second.first,
tree->getNestedModelIndex(mId, current->first));
tree->setData(index, current->second.second);
}
}
}
CSMWorld::CreateCommand::CreateCommand (IdTable& model, const std::string& id, QUndoCommand* parent)
2014-01-27 14:02:29 +00:00
: QUndoCommand (parent), mModel (model), mId (id), mType (UniversalId::Type_None)
{
2014-01-27 14:02:29 +00:00
setText (("Create record " + id).c_str());
}
void CSMWorld::CreateCommand::addValue (int column, const QVariant& value)
{
mValues[column] = value;
}
void CSMWorld::CreateCommand::addNestedValue(int parentColumn, int nestedColumn, const QVariant &value)
{
mNestedValues[parentColumn] = std::make_pair(nestedColumn, value);
}
void CSMWorld::CreateCommand::setType (UniversalId::Type type)
{
mType = type;
}
void CSMWorld::CreateCommand::redo()
{
mModel.addRecordWithData (mId, mValues, mType);
applyModifications();
}
void CSMWorld::CreateCommand::undo()
{
mModel.removeRow (mModel.getModelIndex (mId, 0).row());
2012-12-06 13:56:04 +00:00
}
CSMWorld::RevertCommand::RevertCommand (IdTable& model, const std::string& id, QUndoCommand* parent)
2014-01-27 14:02:29 +00:00
: QUndoCommand (parent), mModel (model), mId (id), mOld (0)
2012-12-06 13:56:04 +00:00
{
2014-01-27 14:02:29 +00:00
setText (("Revert record " + id).c_str());
2012-12-06 13:56:04 +00:00
mOld = model.getRecord (id).clone();
2012-12-06 13:56:04 +00:00
}
CSMWorld::RevertCommand::~RevertCommand()
{
delete mOld;
}
void CSMWorld::RevertCommand::redo()
{
int column = mModel.findColumnIndex (Columns::ColumnId_Modification);
2013-08-26 10:25:52 +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
2014-01-27 14:02:29 +00:00
if (state==RecordBase::State_ModifiedOnly)
2012-12-06 13:56:04 +00:00
{
mModel.removeRows (index.row(), 1);
2012-12-06 13:56:04 +00:00
}
else
{
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
{
mModel.setRecord (mId, *mOld);
2012-12-06 14:18:41 +00:00
}
CSMWorld::DeleteCommand::DeleteCommand (IdTable& model,
const std::string& id, CSMWorld::UniversalId::Type type, QUndoCommand* parent)
: QUndoCommand (parent), mModel (model), mId (id), mOld (0), mType(type)
2012-12-06 14:18:41 +00:00
{
2014-01-27 14:02:29 +00:00
setText (("Delete record " + id).c_str());
2012-12-06 14:18:41 +00:00
mOld = model.getRecord (id).clone();
2012-12-06 14:18:41 +00:00
}
CSMWorld::DeleteCommand::~DeleteCommand()
{
delete mOld;
}
void CSMWorld::DeleteCommand::redo()
{
int column = mModel.findColumnIndex (Columns::ColumnId_Modification);
2013-08-26 10:25:52 +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
2014-01-27 14:02:29 +00:00
if (state==RecordBase::State_ModifiedOnly)
2012-12-06 14:18:41 +00:00
{
mModel.removeRows (index.row(), 1);
2012-12-06 14:18:41 +00:00
}
else
{
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
{
mModel.setRecord (mId, *mOld, mType);
}
CSMWorld::ReorderRowsCommand::ReorderRowsCommand (IdTable& model, int baseIndex,
const std::vector<int>& newOrder)
2014-01-27 14:02:29 +00:00
: mModel (model), mBaseIndex (baseIndex), mNewOrder (newOrder)
{}
void CSMWorld::ReorderRowsCommand::redo()
{
mModel.reorderRows (mBaseIndex, mNewOrder);
}
void CSMWorld::ReorderRowsCommand::undo()
{
int size = static_cast<int> (mNewOrder.size());
std::vector<int> reverse (size);
2014-01-27 14:03:29 +00:00
for (int i=0; i< size; ++i)
reverse.at (mNewOrder[i]) = i;
mModel.reorderRows (mBaseIndex, reverse);
}
2014-01-27 13:46:58 +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
{
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 (mIdOrigin, mId, mType);
applyModifications();
2014-01-27 13:46:58 +00:00
}
void CSMWorld::CloneCommand::undo()
{
mModel.removeRow (mModel.getModelIndex (mId, 0).row());
}
CSMWorld::UpdateCellCommand::UpdateCellCommand (IdTable& model, int row, QUndoCommand *parent)
: QUndoCommand (parent), mModel (model), mRow (row)
{
setText ("Update cell ID");
}
void CSMWorld::UpdateCellCommand::redo()
{
if (!mNew.isValid())
{
int cellColumn = mModel.searchColumnIndex (Columns::ColumnId_Cell);
mIndex = mModel.index (mRow, cellColumn);
const int cellSize = 8192;
QModelIndex xIndex = mModel.index (
mRow, mModel.findColumnIndex (Columns::ColumnId_PositionXPos));
QModelIndex yIndex = mModel.index (
mRow, mModel.findColumnIndex (Columns::ColumnId_PositionYPos));
int x = std::floor (mModel.data (xIndex).toFloat() / cellSize);
int y = std::floor (mModel.data (yIndex).toFloat() / cellSize);
std::ostringstream stream;
stream << "#" << x << " " << y;
mNew = QString::fromUtf8 (stream.str().c_str());
}
mModel.setData (mIndex, mNew);
}
void CSMWorld::UpdateCellCommand::undo()
{
mModel.setData (mIndex, mOld);
}
CSMWorld::DeleteNestedCommand::DeleteNestedCommand (IdTree& model,
2014-07-30 15:07:11 +00:00
const std::string& id,
int nestedRow,
int parentColumn,
2014-07-30 15:07:11 +00:00
QUndoCommand* parent) :
QUndoCommand(parent),
NestedTableStoring(model, id, parentColumn),
2014-07-30 15:07:11 +00:00
mModel(model),
mId(id),
2014-07-30 15:07:11 +00:00
mParentColumn(parentColumn),
mNestedRow(nestedRow)
{
2015-04-11 01:26:29 +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);
mModel.removeRows (mNestedRow, 1, parentIndex);
mModifyParentCommand->redo();
}
void CSMWorld::DeleteNestedCommand::undo()
{
QModelIndex parentIndex = mModel.getModelIndex(mId, mParentColumn);
mModel.setNestedTable(parentIndex, getOld());
mModifyParentCommand->undo();
}
2014-06-24 17:19:40 +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),
2014-06-24 17:19:40 +00:00
mId(id),
mNewRow(nestedRow),
mParentColumn(parentColumn)
2014-06-24 17:19:40 +00:00
{
2015-04-11 01:26:29 +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);
2014-06-24 17:19:40 +00:00
mModel.addNestedRow (parentIndex, mNewRow);
mModifyParentCommand->redo();
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)
: 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;
}