2012-11-29 18:09:06 +00:00
|
|
|
|
|
|
|
#include "table.hpp"
|
|
|
|
|
|
|
|
#include <QHeaderView>
|
2012-12-03 20:44:16 +00:00
|
|
|
#include <QAction>
|
2014-02-20 15:55:51 +00:00
|
|
|
#include <QApplication>
|
2012-12-03 20:44:16 +00:00
|
|
|
#include <QMenu>
|
|
|
|
#include <QContextMenuEvent>
|
2014-02-13 18:00:35 +00:00
|
|
|
#include <QString>
|
2014-02-16 16:22:46 +00:00
|
|
|
#include <QtCore/qnamespace.h>
|
2012-11-29 18:09:06 +00:00
|
|
|
|
2014-03-06 12:51:21 +00:00
|
|
|
#include "../../model/doc/document.hpp"
|
|
|
|
|
2012-11-29 18:09:06 +00:00
|
|
|
#include "../../model/world/data.hpp"
|
|
|
|
#include "../../model/world/commands.hpp"
|
2012-12-03 20:44:16 +00:00
|
|
|
#include "../../model/world/idtableproxymodel.hpp"
|
2012-12-06 13:56:04 +00:00
|
|
|
#include "../../model/world/idtable.hpp"
|
|
|
|
#include "../../model/world/record.hpp"
|
2013-11-14 10:39:14 +00:00
|
|
|
#include "../../model/world/columns.hpp"
|
2014-02-04 08:13:40 +00:00
|
|
|
#include "../../model/world/tablemimedata.hpp"
|
2014-02-14 13:04:36 +00:00
|
|
|
#include "../../model/world/tablemimedata.hpp"
|
2013-09-22 11:39:44 +00:00
|
|
|
|
2013-06-15 11:40:18 +00:00
|
|
|
#include "recordstatusdelegate.hpp"
|
2012-12-13 10:24:39 +00:00
|
|
|
#include "util.hpp"
|
2012-11-29 18:09:06 +00:00
|
|
|
|
2012-12-06 14:25:31 +00:00
|
|
|
void CSVWorld::Table::contextMenuEvent (QContextMenuEvent *event)
|
2012-12-03 20:44:16 +00:00
|
|
|
{
|
2012-12-06 13:56:04 +00:00
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
QMenu menu (this);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
/// \todo add menu items for select all and clear selection
|
2012-12-03 20:44:16 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
if (!mEditLock)
|
|
|
|
{
|
2013-07-21 15:53:39 +00:00
|
|
|
if (selectedRows.size()==1)
|
2014-01-14 14:44:04 +00:00
|
|
|
{
|
2013-07-21 15:53:39 +00:00
|
|
|
menu.addAction (mEditAction);
|
2014-03-02 21:43:44 +00:00
|
|
|
|
2014-01-14 14:44:04 +00:00
|
|
|
if (mCreateAction)
|
|
|
|
menu.addAction(mCloneAction);
|
|
|
|
}
|
2013-07-21 15:53:39 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
if (mCreateAction)
|
|
|
|
menu.addAction (mCreateAction);
|
2012-12-06 13:56:04 +00:00
|
|
|
|
2013-11-14 10:55:02 +00:00
|
|
|
/// \todo Reverting temporarily disabled on tables that support reordering, because
|
|
|
|
/// revert logic currently can not handle reordering.
|
|
|
|
if (mModel->getReordering()==CSMWorld::IdTable::Reordering_None)
|
|
|
|
if (listRevertableSelectedIds().size()>0)
|
|
|
|
menu.addAction (mRevertAction);
|
2012-12-08 17:15:00 +00:00
|
|
|
|
|
|
|
if (listDeletableSelectedIds().size()>0)
|
|
|
|
menu.addAction (mDeleteAction);
|
2013-11-14 10:39:14 +00:00
|
|
|
|
|
|
|
if (mModel->getReordering()==CSMWorld::IdTable::Reordering_WithinTopic)
|
|
|
|
{
|
|
|
|
/// \todo allow reordering of multiple rows
|
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
|
|
|
int row =selectedRows.begin()->row();
|
|
|
|
|
2013-11-16 22:08:03 +00:00
|
|
|
int column = mModel->searchColumnIndex (CSMWorld::Columns::ColumnId_Topic);
|
2013-11-14 10:39:14 +00:00
|
|
|
|
|
|
|
if (column==-1)
|
2013-11-16 22:08:03 +00:00
|
|
|
column = mModel->searchColumnIndex (CSMWorld::Columns::ColumnId_Journal);
|
2013-11-14 10:39:14 +00:00
|
|
|
|
|
|
|
if (column!=-1)
|
|
|
|
{
|
|
|
|
if (row>0 && mProxyModel->data (mProxyModel->index (row, column))==
|
|
|
|
mProxyModel->data (mProxyModel->index (row-1, column)))
|
|
|
|
{
|
|
|
|
menu.addAction (mMoveUpAction);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (row<mProxyModel->rowCount()-1 && mProxyModel->data (mProxyModel->index (row, column))==
|
|
|
|
mProxyModel->data (mProxyModel->index (row+1, column)))
|
|
|
|
{
|
|
|
|
menu.addAction (mMoveDownAction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-12-08 17:15:00 +00:00
|
|
|
}
|
2012-12-06 14:18:41 +00:00
|
|
|
|
2014-03-10 11:44:34 +00:00
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
|
|
|
if (mModel->getViewing()!=CSMWorld::IdTable::Viewing_None)
|
|
|
|
{
|
|
|
|
int row = selectedRows.begin()->row();
|
|
|
|
|
|
|
|
row = mProxyModel->mapToSource (mProxyModel->index (row, 0)).row();
|
|
|
|
|
|
|
|
CSMWorld::UniversalId id = mModel->view (row).first;
|
|
|
|
|
|
|
|
int index = mDocument.getData().getCells().searchId (id.getId());
|
|
|
|
// index==-1: the ID references a worldspace instead of a cell (ignore for now and go
|
|
|
|
// ahead)
|
|
|
|
|
|
|
|
if (index==-1 || !mDocument.getData().getCells().getRecord (index).isDeleted())
|
|
|
|
menu.addAction (mViewAction);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mModel->hasPreview())
|
|
|
|
menu.addAction (mPreviewAction);
|
|
|
|
}
|
|
|
|
|
2012-12-03 20:44:16 +00:00
|
|
|
menu.exec (event->globalPos());
|
|
|
|
}
|
|
|
|
|
2012-12-06 14:25:31 +00:00
|
|
|
std::vector<std::string> CSVWorld::Table::listRevertableSelectedIds() const
|
|
|
|
{
|
|
|
|
std::vector<std::string> revertableIds;
|
|
|
|
|
2013-07-25 12:38:53 +00:00
|
|
|
if (mProxyModel->columnCount()>0)
|
2012-12-06 14:25:31 +00:00
|
|
|
{
|
2013-07-25 12:38:53 +00:00
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
for (QModelIndexList::const_iterator iter (selectedRows.begin()); iter!=selectedRows.end();
|
2014-02-10 15:48:04 +00:00
|
|
|
++iter)
|
2013-07-25 12:38:53 +00:00
|
|
|
{
|
|
|
|
QModelIndex index = mProxyModel->mapToSource (mProxyModel->index (iter->row(), 0));
|
|
|
|
|
|
|
|
CSMWorld::RecordBase::State state =
|
|
|
|
static_cast<CSMWorld::RecordBase::State> (
|
|
|
|
mModel->data (mModel->index (index.row(), 1)).toInt());
|
2012-12-06 14:25:31 +00:00
|
|
|
|
2013-07-25 12:38:53 +00:00
|
|
|
if (state!=CSMWorld::RecordBase::State_BaseOnly)
|
|
|
|
{
|
2013-08-08 10:49:30 +00:00
|
|
|
int columnIndex = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
|
|
|
|
std::string id = mModel->data (mModel->index (index.row(), columnIndex)).
|
2013-07-25 12:38:53 +00:00
|
|
|
toString().toUtf8().constData();
|
2012-12-06 14:25:31 +00:00
|
|
|
|
2013-07-25 12:38:53 +00:00
|
|
|
revertableIds.push_back (id);
|
|
|
|
}
|
|
|
|
}
|
2012-12-06 14:25:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return revertableIds;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> CSVWorld::Table::listDeletableSelectedIds() const
|
|
|
|
{
|
|
|
|
std::vector<std::string> deletableIds;
|
|
|
|
|
2013-07-25 12:38:53 +00:00
|
|
|
if (mProxyModel->columnCount()>0)
|
2012-12-06 14:25:31 +00:00
|
|
|
{
|
2013-07-25 12:38:53 +00:00
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
for (QModelIndexList::const_iterator iter (selectedRows.begin()); iter!=selectedRows.end();
|
|
|
|
++iter)
|
|
|
|
{
|
|
|
|
QModelIndex index = mProxyModel->mapToSource (mProxyModel->index (iter->row(), 0));
|
|
|
|
|
2013-10-21 11:58:47 +00:00
|
|
|
// check record state
|
2013-07-25 12:38:53 +00:00
|
|
|
CSMWorld::RecordBase::State state =
|
|
|
|
static_cast<CSMWorld::RecordBase::State> (
|
|
|
|
mModel->data (mModel->index (index.row(), 1)).toInt());
|
2012-12-06 14:25:31 +00:00
|
|
|
|
2013-10-21 11:58:47 +00:00
|
|
|
if (state==CSMWorld::RecordBase::State_Deleted)
|
|
|
|
continue;
|
2013-08-08 10:49:30 +00:00
|
|
|
|
2013-10-21 11:58:47 +00:00
|
|
|
// check other columns (only relevant for a subset of the tables)
|
|
|
|
int dialogueTypeIndex =
|
|
|
|
mModel->searchColumnIndex (CSMWorld::Columns::ColumnId_DialogueType);
|
2012-12-06 14:25:31 +00:00
|
|
|
|
2013-10-21 11:58:47 +00:00
|
|
|
if (dialogueTypeIndex!=-1)
|
|
|
|
{
|
|
|
|
int type = mModel->data (mModel->index (index.row(), dialogueTypeIndex)).toInt();
|
|
|
|
|
|
|
|
if (type!=ESM::Dialogue::Topic && type!=ESM::Dialogue::Journal)
|
|
|
|
continue;
|
2013-07-25 12:38:53 +00:00
|
|
|
}
|
2013-10-21 11:58:47 +00:00
|
|
|
|
|
|
|
// add the id to the collection
|
|
|
|
int columnIndex = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
|
|
|
|
std::string id = mModel->data (mModel->index (index.row(), columnIndex)).
|
|
|
|
toString().toUtf8().constData();
|
|
|
|
|
|
|
|
deletableIds.push_back (id);
|
2013-07-25 12:38:53 +00:00
|
|
|
}
|
2012-12-06 14:25:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return deletableIds;
|
|
|
|
}
|
|
|
|
|
2014-03-06 12:51:21 +00:00
|
|
|
CSVWorld::Table::Table (const CSMWorld::UniversalId& id,
|
|
|
|
bool createAndDelete, bool sorting, CSMDoc::Document& document)
|
|
|
|
: mCreateAction (0), mCloneAction(0), mEditLock (false), mRecordStatusDisplay (0),
|
|
|
|
mDocument (document)
|
2012-11-29 18:09:06 +00:00
|
|
|
{
|
2014-03-06 12:51:21 +00:00
|
|
|
mModel = &dynamic_cast<CSMWorld::IdTable&> (*mDocument.getData().getTableModel (id));
|
2012-11-29 18:09:06 +00:00
|
|
|
|
2012-12-06 13:56:04 +00:00
|
|
|
mProxyModel = new CSMWorld::IdTableProxyModel (this);
|
|
|
|
mProxyModel->setSourceModel (mModel);
|
2012-11-29 18:09:06 +00:00
|
|
|
|
2012-12-06 13:56:04 +00:00
|
|
|
setModel (mProxyModel);
|
2012-11-29 18:09:06 +00:00
|
|
|
horizontalHeader()->setResizeMode (QHeaderView::Interactive);
|
|
|
|
verticalHeader()->hide();
|
2013-10-31 12:40:14 +00:00
|
|
|
setSortingEnabled (sorting);
|
2012-11-29 18:09:06 +00:00
|
|
|
setSelectionBehavior (QAbstractItemView::SelectRows);
|
|
|
|
setSelectionMode (QAbstractItemView::ExtendedSelection);
|
|
|
|
|
2012-12-13 14:03:35 +00:00
|
|
|
int columns = mModel->columnCount();
|
|
|
|
|
|
|
|
for (int i=0; i<columns; ++i)
|
|
|
|
{
|
|
|
|
int flags = mModel->headerData (i, Qt::Horizontal, CSMWorld::ColumnBase::Role_Flags).toInt();
|
|
|
|
|
|
|
|
if (flags & CSMWorld::ColumnBase::Flag_Table)
|
|
|
|
{
|
2013-02-10 16:21:25 +00:00
|
|
|
CSMWorld::ColumnBase::Display display = static_cast<CSMWorld::ColumnBase::Display> (
|
|
|
|
mModel->headerData (i, Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
|
|
|
|
|
|
|
CommandDelegate *delegate = CommandDelegateFactoryCollection::get().makeDelegate (display,
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack(), this);
|
2013-02-10 16:21:25 +00:00
|
|
|
|
2012-12-13 14:03:35 +00:00
|
|
|
mDelegates.push_back (delegate);
|
|
|
|
setItemDelegateForColumn (i, delegate);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
hideColumn (i);
|
|
|
|
}
|
|
|
|
|
2013-07-21 15:53:39 +00:00
|
|
|
mEditAction = new QAction (tr ("Edit Record"), this);
|
|
|
|
connect (mEditAction, SIGNAL (triggered()), this, SLOT (editRecord()));
|
|
|
|
addAction (mEditAction);
|
2012-12-03 20:44:16 +00:00
|
|
|
|
|
|
|
if (createAndDelete)
|
|
|
|
{
|
2012-12-03 21:03:02 +00:00
|
|
|
mCreateAction = new QAction (tr ("Add Record"), this);
|
2013-07-26 16:22:31 +00:00
|
|
|
connect (mCreateAction, SIGNAL (triggered()), this, SIGNAL (createRequest()));
|
2012-12-03 20:44:16 +00:00
|
|
|
addAction (mCreateAction);
|
2014-01-27 18:40:05 +00:00
|
|
|
|
2014-01-14 14:44:04 +00:00
|
|
|
mCloneAction = new QAction (tr ("Clone Record"), this);
|
|
|
|
connect(mCloneAction, SIGNAL (triggered()), this, SLOT (cloneRecord()));
|
2014-01-14 12:12:15 +00:00
|
|
|
addAction(mCloneAction);
|
2012-12-03 20:44:16 +00:00
|
|
|
}
|
2012-12-06 13:56:04 +00:00
|
|
|
|
|
|
|
mRevertAction = new QAction (tr ("Revert Record"), this);
|
|
|
|
connect (mRevertAction, SIGNAL (triggered()), this, SLOT (revertRecord()));
|
|
|
|
addAction (mRevertAction);
|
2012-12-06 14:18:41 +00:00
|
|
|
|
|
|
|
mDeleteAction = new QAction (tr ("Delete Record"), this);
|
|
|
|
connect (mDeleteAction, SIGNAL (triggered()), this, SLOT (deleteRecord()));
|
|
|
|
addAction (mDeleteAction);
|
2013-07-25 12:29:56 +00:00
|
|
|
|
2013-11-14 10:39:14 +00:00
|
|
|
mMoveUpAction = new QAction (tr ("Move Up"), this);
|
|
|
|
connect (mMoveUpAction, SIGNAL (triggered()), this, SLOT (moveUpRecord()));
|
|
|
|
addAction (mMoveUpAction);
|
|
|
|
|
|
|
|
mMoveDownAction = new QAction (tr ("Move Down"), this);
|
|
|
|
connect (mMoveDownAction, SIGNAL (triggered()), this, SLOT (moveDownRecord()));
|
|
|
|
addAction (mMoveDownAction);
|
|
|
|
|
2014-03-02 21:43:44 +00:00
|
|
|
mViewAction = new QAction (tr ("View"), this);
|
|
|
|
connect (mViewAction, SIGNAL (triggered()), this, SLOT (viewRecord()));
|
|
|
|
addAction (mViewAction);
|
|
|
|
|
2014-03-10 11:44:34 +00:00
|
|
|
mPreviewAction = new QAction (tr ("Preview"), this);
|
|
|
|
connect (mPreviewAction, SIGNAL (triggered()), this, SLOT (previewRecord()));
|
|
|
|
addAction (mPreviewAction);
|
|
|
|
|
2013-07-25 12:29:56 +00:00
|
|
|
connect (mProxyModel, SIGNAL (rowsInserted (const QModelIndex&, int, int)),
|
|
|
|
this, SLOT (tableSizeUpdate()));
|
|
|
|
|
|
|
|
/// \note This signal could instead be connected to a slot that filters out changes not affecting
|
|
|
|
/// the records status column (for permanence reasons)
|
|
|
|
connect (mProxyModel, SIGNAL (dataChanged (const QModelIndex&, const QModelIndex&)),
|
|
|
|
this, SLOT (tableSizeUpdate()));
|
|
|
|
|
|
|
|
connect (selectionModel(), SIGNAL (selectionChanged (const QItemSelection&, const QItemSelection&)),
|
|
|
|
this, SLOT (selectionSizeUpdate ()));
|
2014-02-10 15:48:04 +00:00
|
|
|
|
|
|
|
setAcceptDrops(true);
|
2012-11-30 12:58:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::setEditLock (bool locked)
|
|
|
|
{
|
|
|
|
for (std::vector<CommandDelegate *>::iterator iter (mDelegates.begin()); iter!=mDelegates.end(); ++iter)
|
|
|
|
(*iter)->setEditLock (locked);
|
2012-12-08 17:15:00 +00:00
|
|
|
|
|
|
|
mEditLock = locked;
|
2012-12-03 20:44:16 +00:00
|
|
|
}
|
|
|
|
|
2012-12-13 12:35:08 +00:00
|
|
|
CSMWorld::UniversalId CSVWorld::Table::getUniversalId (int row) const
|
|
|
|
{
|
|
|
|
return CSMWorld::UniversalId (
|
|
|
|
static_cast<CSMWorld::UniversalId::Type> (mProxyModel->data (mProxyModel->index (row, 2)).toInt()),
|
2014-03-21 10:56:48 +00:00
|
|
|
mProxyModel->data (mProxyModel->index (row, 0)).toString().toUtf8().constData());
|
2012-12-13 12:35:08 +00:00
|
|
|
}
|
|
|
|
|
2012-12-06 13:56:04 +00:00
|
|
|
void CSVWorld::Table::revertRecord()
|
|
|
|
{
|
2012-12-08 17:15:00 +00:00
|
|
|
if (!mEditLock)
|
2012-12-06 13:56:04 +00:00
|
|
|
{
|
2012-12-08 17:15:00 +00:00
|
|
|
std::vector<std::string> revertableIds = listRevertableSelectedIds();
|
|
|
|
|
2014-04-27 21:24:25 +00:00
|
|
|
if (!revertableIds.empty())
|
2012-12-08 17:15:00 +00:00
|
|
|
{
|
|
|
|
if (revertableIds.size()>1)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().beginMacro (tr ("Revert multiple records"));
|
2012-12-06 13:56:04 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
for (std::vector<std::string>::const_iterator iter (revertableIds.begin()); iter!=revertableIds.end(); ++iter)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().push (new CSMWorld::RevertCommand (*mModel, *iter));
|
2012-12-06 13:56:04 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
if (revertableIds.size()>1)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().endMacro();
|
2012-12-08 17:15:00 +00:00
|
|
|
}
|
2012-12-06 14:18:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::deleteRecord()
|
|
|
|
{
|
2012-12-08 17:15:00 +00:00
|
|
|
if (!mEditLock)
|
2012-12-06 14:18:41 +00:00
|
|
|
{
|
2012-12-08 17:15:00 +00:00
|
|
|
std::vector<std::string> deletableIds = listDeletableSelectedIds();
|
|
|
|
|
2014-04-27 21:24:25 +00:00
|
|
|
if (!deletableIds.empty())
|
2012-12-08 17:15:00 +00:00
|
|
|
{
|
|
|
|
if (deletableIds.size()>1)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().beginMacro (tr ("Delete multiple records"));
|
2012-12-06 14:18:41 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
for (std::vector<std::string>::const_iterator iter (deletableIds.begin()); iter!=deletableIds.end(); ++iter)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().push (new CSMWorld::DeleteCommand (*mModel, *iter));
|
2012-12-06 14:18:41 +00:00
|
|
|
|
2012-12-08 17:15:00 +00:00
|
|
|
if (deletableIds.size()>1)
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().endMacro();
|
2012-12-08 17:15:00 +00:00
|
|
|
}
|
2012-12-06 13:56:04 +00:00
|
|
|
}
|
2013-06-15 11:40:18 +00:00
|
|
|
}
|
|
|
|
|
2013-07-21 15:53:39 +00:00
|
|
|
void CSVWorld::Table::editRecord()
|
|
|
|
{
|
|
|
|
if (!mEditLock)
|
|
|
|
{
|
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size()==1)
|
2014-03-02 21:34:41 +00:00
|
|
|
emit editRequest (getUniversalId (selectedRows.begin()->row()), "");
|
2013-07-21 15:53:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-14 14:44:04 +00:00
|
|
|
void CSVWorld::Table::cloneRecord()
|
|
|
|
{
|
|
|
|
if (!mEditLock)
|
|
|
|
{
|
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
2014-01-21 07:27:29 +00:00
|
|
|
const CSMWorld::UniversalId& toClone = getUniversalId(selectedRows.begin()->row());
|
|
|
|
if (selectedRows.size()==1 && !mModel->getRecord(toClone.getId()).isDeleted())
|
2014-01-20 14:58:19 +00:00
|
|
|
{
|
2014-01-21 07:27:29 +00:00
|
|
|
emit cloneRequest (toClone);
|
2014-01-20 14:58:19 +00:00
|
|
|
}
|
2014-01-14 14:44:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-14 10:39:14 +00:00
|
|
|
void CSVWorld::Table::moveUpRecord()
|
|
|
|
{
|
2014-04-12 18:07:09 +00:00
|
|
|
if (mEditLock)
|
|
|
|
return;
|
|
|
|
|
2013-11-14 10:39:14 +00:00
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
|
|
|
int row2 =selectedRows.begin()->row();
|
|
|
|
|
|
|
|
if (row2>0)
|
|
|
|
{
|
|
|
|
int row = row2-1;
|
|
|
|
|
|
|
|
row = mProxyModel->mapToSource (mProxyModel->index (row, 0)).row();
|
|
|
|
row2 = mProxyModel->mapToSource (mProxyModel->index (row2, 0)).row();
|
|
|
|
|
|
|
|
if (row2<=row)
|
|
|
|
throw std::runtime_error ("Inconsistent row order");
|
|
|
|
|
|
|
|
std::vector<int> newOrder (row2-row+1);
|
|
|
|
newOrder[0] = row2-row;
|
|
|
|
newOrder[row2-row] = 0;
|
|
|
|
for (int i=1; i<row2-row; ++i)
|
|
|
|
newOrder[i] = i;
|
|
|
|
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().push (new CSMWorld::ReorderRowsCommand (*mModel, row, newOrder));
|
2013-11-14 10:39:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::moveDownRecord()
|
|
|
|
{
|
2014-04-12 18:07:09 +00:00
|
|
|
if (mEditLock)
|
|
|
|
return;
|
|
|
|
|
2013-11-14 10:39:14 +00:00
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
|
|
|
int row =selectedRows.begin()->row();
|
|
|
|
|
|
|
|
if (row<mProxyModel->rowCount()-1)
|
|
|
|
{
|
|
|
|
int row2 = row+1;
|
|
|
|
|
|
|
|
row = mProxyModel->mapToSource (mProxyModel->index (row, 0)).row();
|
|
|
|
row2 = mProxyModel->mapToSource (mProxyModel->index (row2, 0)).row();
|
|
|
|
|
|
|
|
if (row2<=row)
|
|
|
|
throw std::runtime_error ("Inconsistent row order");
|
|
|
|
|
|
|
|
std::vector<int> newOrder (row2-row+1);
|
|
|
|
newOrder[0] = row2-row;
|
|
|
|
newOrder[row2-row] = 0;
|
|
|
|
for (int i=1; i<row2-row; ++i)
|
|
|
|
newOrder[i] = i;
|
|
|
|
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().push (new CSMWorld::ReorderRowsCommand (*mModel, row, newOrder));
|
2013-11-14 10:39:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-02 21:43:44 +00:00
|
|
|
void CSVWorld::Table::viewRecord()
|
|
|
|
{
|
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
2014-03-10 11:44:34 +00:00
|
|
|
int row = selectedRows.begin()->row();
|
2014-03-02 21:43:44 +00:00
|
|
|
|
|
|
|
row = mProxyModel->mapToSource (mProxyModel->index (row, 0)).row();
|
|
|
|
|
|
|
|
std::pair<CSMWorld::UniversalId, std::string> params = mModel->view (row);
|
|
|
|
|
|
|
|
if (params.first.getType()!=CSMWorld::UniversalId::Type_None)
|
|
|
|
emit editRequest (params.first, params.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-10 11:44:34 +00:00
|
|
|
void CSVWorld::Table::previewRecord()
|
|
|
|
{
|
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size()==1)
|
|
|
|
{
|
|
|
|
std::string id = getUniversalId (selectedRows.begin()->row()).getId();
|
|
|
|
|
|
|
|
emit editRequest (CSMWorld::UniversalId (CSMWorld::UniversalId::Type_Preview, id) , "");
|
|
|
|
}
|
|
|
|
}
|
2014-03-19 10:42:43 +00:00
|
|
|
|
2014-04-23 03:19:53 +00:00
|
|
|
void CSVWorld::Table::updateUserSetting
|
|
|
|
(const QString &name, const QStringList &list)
|
2013-06-15 11:40:18 +00:00
|
|
|
{
|
2013-07-20 11:19:27 +00:00
|
|
|
int columns = mModel->columnCount();
|
2013-07-11 02:13:59 +00:00
|
|
|
|
2013-07-20 11:19:27 +00:00
|
|
|
for (int i=0; i<columns; ++i)
|
|
|
|
if (QAbstractItemDelegate *delegate = itemDelegateForColumn (i))
|
2014-04-24 17:50:10 +00:00
|
|
|
{
|
|
|
|
dynamic_cast<CommandDelegate&>
|
|
|
|
(*delegate).updateUserSetting (name, list);
|
2014-04-23 03:19:53 +00:00
|
|
|
{
|
|
|
|
emit dataChanged (mModel->index (0, i),
|
|
|
|
mModel->index (mModel->rowCount()-1, i));
|
|
|
|
}
|
2014-04-24 17:50:10 +00:00
|
|
|
}
|
2013-06-15 11:40:18 +00:00
|
|
|
}
|
2013-07-25 12:29:56 +00:00
|
|
|
|
|
|
|
void CSVWorld::Table::tableSizeUpdate()
|
|
|
|
{
|
|
|
|
int size = 0;
|
|
|
|
int deleted = 0;
|
|
|
|
int modified = 0;
|
|
|
|
|
2013-08-23 12:11:33 +00:00
|
|
|
if (mProxyModel->columnCount()>0)
|
2013-07-25 12:29:56 +00:00
|
|
|
{
|
2013-08-23 12:11:33 +00:00
|
|
|
int rows = mProxyModel->rowCount();
|
2013-07-25 12:29:56 +00:00
|
|
|
|
|
|
|
for (int i=0; i<rows; ++i)
|
|
|
|
{
|
|
|
|
QModelIndex index = mProxyModel->mapToSource (mProxyModel->index (i, 0));
|
|
|
|
|
2013-08-08 10:49:30 +00:00
|
|
|
int columnIndex = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Modification);
|
|
|
|
int state = mModel->data (mModel->index (index.row(), columnIndex)).toInt();
|
2013-07-25 12:29:56 +00:00
|
|
|
|
|
|
|
switch (state)
|
|
|
|
{
|
2014-02-14 11:46:15 +00:00
|
|
|
case CSMWorld::RecordBase::State_BaseOnly: ++size; break;
|
|
|
|
case CSMWorld::RecordBase::State_Modified: ++size; ++modified; break;
|
|
|
|
case CSMWorld::RecordBase::State_ModifiedOnly: ++size; ++modified; break;
|
|
|
|
case CSMWorld::RecordBase:: State_Deleted: ++deleted; ++modified; break;
|
2013-07-25 12:29:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tableSizeChanged (size, deleted, modified);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::selectionSizeUpdate()
|
|
|
|
{
|
|
|
|
selectionSizeChanged (selectionModel()->selectedRows().size());
|
2013-07-29 13:00:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::requestFocus (const std::string& id)
|
|
|
|
{
|
|
|
|
QModelIndex index = mProxyModel->getModelIndex (id, 0);
|
|
|
|
|
|
|
|
if (index.isValid())
|
|
|
|
scrollTo (index, QAbstractItemView::PositionAtTop);
|
2013-08-18 14:53:28 +00:00
|
|
|
}
|
|
|
|
|
2013-08-22 07:17:57 +00:00
|
|
|
void CSVWorld::Table::recordFilterChanged (boost::shared_ptr<CSMFilter::Node> filter)
|
2013-08-18 14:53:28 +00:00
|
|
|
{
|
2013-08-22 07:17:57 +00:00
|
|
|
mProxyModel->setFilter (filter);
|
2013-11-16 22:08:03 +00:00
|
|
|
}
|
2014-02-04 10:40:48 +00:00
|
|
|
|
|
|
|
void CSVWorld::Table::mouseMoveEvent (QMouseEvent* event)
|
|
|
|
{
|
2014-02-05 10:44:08 +00:00
|
|
|
if (event->buttons() & Qt::LeftButton)
|
|
|
|
{
|
|
|
|
QModelIndexList selectedRows = selectionModel()->selectedRows();
|
|
|
|
|
|
|
|
if (selectedRows.size() == 0)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDrag* drag = new QDrag (this);
|
|
|
|
CSMWorld::TableMimeData* mime = NULL;
|
|
|
|
|
|
|
|
if (selectedRows.size() == 1)
|
|
|
|
{
|
2014-02-15 11:40:07 +00:00
|
|
|
mime = new CSMWorld::TableMimeData (getUniversalId (selectedRows.begin()->row()), mDocument);
|
2014-02-05 10:44:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::vector<CSMWorld::UniversalId> idToDrag;
|
|
|
|
|
2014-02-11 11:25:21 +00:00
|
|
|
foreach (QModelIndex it, selectedRows) //I had a dream. Dream where you could use C++11 in OpenMW.
|
2014-02-05 10:44:08 +00:00
|
|
|
{
|
|
|
|
idToDrag.push_back (getUniversalId (it.row()));
|
|
|
|
}
|
|
|
|
|
2014-02-15 11:40:07 +00:00
|
|
|
mime = new CSMWorld::TableMimeData (idToDrag, mDocument);
|
2014-02-05 10:44:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
drag->setMimeData (mime);
|
2014-03-21 10:56:48 +00:00
|
|
|
drag->setPixmap (QString::fromUtf8 (mime->getIcon().c_str()));
|
2014-03-13 07:52:37 +00:00
|
|
|
drag->exec(Qt::CopyAction);
|
2014-02-10 15:48:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::dragEnterEvent(QDragEnterEvent *event)
|
|
|
|
{
|
2014-03-12 18:36:46 +00:00
|
|
|
event->acceptProposedAction();
|
2014-02-10 15:48:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::dropEvent(QDropEvent *event)
|
|
|
|
{
|
2014-02-13 12:54:09 +00:00
|
|
|
QModelIndex index = indexAt (event->pos());
|
|
|
|
|
2014-02-21 11:55:01 +00:00
|
|
|
if (!index.isValid())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-02-15 12:22:14 +00:00
|
|
|
const CSMWorld::TableMimeData* mime = dynamic_cast<const CSMWorld::TableMimeData*> (event->mimeData());
|
|
|
|
if (mime->fromDocument (mDocument))
|
2014-02-13 12:54:09 +00:00
|
|
|
{
|
2014-02-15 11:40:07 +00:00
|
|
|
CSMWorld::ColumnBase::Display display = static_cast<CSMWorld::ColumnBase::Display>
|
|
|
|
(mModel->headerData (index.column(), Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
2014-02-14 11:46:15 +00:00
|
|
|
|
2014-02-15 12:22:14 +00:00
|
|
|
if (mime->holdsType (display))
|
2014-02-15 11:40:07 +00:00
|
|
|
{
|
|
|
|
CSMWorld::UniversalId record (mime->returnMatching (display));
|
2014-02-14 11:46:15 +00:00
|
|
|
|
2014-02-15 11:40:07 +00:00
|
|
|
std::auto_ptr<CSMWorld::ModifyCommand> command (new CSMWorld::ModifyCommand
|
2014-02-20 13:10:03 +00:00
|
|
|
(*mProxyModel, index, QVariant (QString::fromUtf8 (record.getId().c_str()))));
|
2014-02-14 11:46:15 +00:00
|
|
|
|
2014-03-06 12:51:21 +00:00
|
|
|
mDocument.getUndoStack().push (command.release());
|
2014-02-15 11:40:07 +00:00
|
|
|
}
|
|
|
|
} //TODO handle drops from different document
|
2014-02-12 10:16:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::Table::dragMoveEvent(QDragMoveEvent *event)
|
|
|
|
{
|
2014-03-12 18:36:46 +00:00
|
|
|
event->accept();
|
2014-02-12 12:13:32 +00:00
|
|
|
}
|
2014-02-19 15:15:51 +00:00
|
|
|
|
|
|
|
std::vector<std::string> CSVWorld::Table::getColumnsWithDisplay(CSMWorld::ColumnBase::Display display) const
|
|
|
|
{
|
2014-02-20 11:43:09 +00:00
|
|
|
const int count = mModel->columnCount();
|
2014-02-19 15:15:51 +00:00
|
|
|
|
|
|
|
std::vector<std::string> titles;
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
CSMWorld::ColumnBase::Display columndisplay = static_cast<CSMWorld::ColumnBase::Display>
|
|
|
|
(mModel->headerData (i, Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
|
|
|
|
|
|
|
if (display == columndisplay)
|
|
|
|
{
|
2014-03-21 10:56:48 +00:00
|
|
|
titles.push_back(mModel->headerData (i, Qt::Horizontal).toString().toUtf8().constData());
|
2014-02-19 15:15:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return titles;
|
2014-04-23 03:19:53 +00:00
|
|
|
}
|