2012-11-26 12:29:22 +01:00
|
|
|
#include "idtable.hpp"
|
2014-07-07 12:07:29 +02:00
|
|
|
#include <QDebug>
|
2012-11-26 12:29:22 +01:00
|
|
|
|
2014-06-19 18:46:09 +02:00
|
|
|
#include <cassert>
|
2014-07-18 18:26:22 +02:00
|
|
|
#include "nestedtablewrapper.hpp"
|
2014-06-19 18:46:09 +02:00
|
|
|
|
2013-06-17 11:42:02 +02:00
|
|
|
#include "collectionbase.hpp"
|
|
|
|
#include "columnbase.hpp"
|
2012-11-26 12:29:22 +01:00
|
|
|
|
2014-06-05 10:28:10 +02:00
|
|
|
CSMWorld::IdTable::IdTable (CollectionBase *idCollection, unsigned int features)
|
2014-07-04 13:24:35 +02:00
|
|
|
: IdTableBase (features), mIdCollection (idCollection)
|
2013-11-14 11:39:14 +01:00
|
|
|
{}
|
2012-11-26 12:29:22 +01:00
|
|
|
|
|
|
|
CSMWorld::IdTable::~IdTable()
|
2013-11-14 11:39:14 +01:00
|
|
|
{}
|
2012-11-26 12:29:22 +01:00
|
|
|
|
|
|
|
int CSMWorld::IdTable::rowCount (const QModelIndex & parent) const
|
|
|
|
{
|
2014-06-09 10:35:39 +02:00
|
|
|
if (hasChildren(parent))
|
|
|
|
{
|
2014-06-17 18:28:49 +02:00
|
|
|
return mIdCollection->getNestedRowsCount(parent.row(), parent.column());
|
2014-06-09 10:35:39 +02:00
|
|
|
}
|
2012-11-26 12:29:22 +01:00
|
|
|
|
|
|
|
return mIdCollection->getSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::columnCount (const QModelIndex & parent) const
|
|
|
|
{
|
2014-06-17 18:28:49 +02:00
|
|
|
if (parent.isValid() &&
|
|
|
|
parent.data().isValid())
|
2014-06-09 10:35:39 +02:00
|
|
|
{
|
2014-06-17 11:49:35 +02:00
|
|
|
return mIdCollection->getNestedColumnsCount(parent.row(), parent.column());
|
2014-06-09 10:35:39 +02:00
|
|
|
}
|
2012-11-26 12:29:22 +01:00
|
|
|
|
2012-11-29 18:56:28 +01:00
|
|
|
return mIdCollection->getColumns();
|
2012-11-26 12:29:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant CSMWorld::IdTable::data (const QModelIndex & index, int role) const
|
|
|
|
{
|
2012-11-29 14:45:34 +01:00
|
|
|
if (role!=Qt::DisplayRole && role!=Qt::EditRole)
|
2012-11-26 12:29:22 +01:00
|
|
|
return QVariant();
|
|
|
|
|
2012-12-13 14:53:16 +01:00
|
|
|
if (role==Qt::EditRole && !mIdCollection->getColumn (index.column()).isEditable())
|
2012-12-11 15:35:47 +01:00
|
|
|
return QVariant();
|
2012-11-26 12:29:22 +01:00
|
|
|
|
2014-06-17 18:28:49 +02:00
|
|
|
if (index.internalId() != 0)
|
2014-06-09 10:35:39 +02:00
|
|
|
{
|
|
|
|
std::pair<int, int> parentAdress(unfoldIndexAdress(index.internalId()));
|
2014-06-17 18:28:49 +02:00
|
|
|
return mIdCollection->getNestedData(parentAdress.first,
|
|
|
|
parentAdress.second,
|
|
|
|
index.row(),
|
|
|
|
index.column());
|
2014-06-09 10:35:39 +02:00
|
|
|
} else {
|
|
|
|
return mIdCollection->getData (index.row(), index.column());
|
|
|
|
}
|
2012-11-26 12:29:22 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 18:28:49 +02:00
|
|
|
QVariant CSMWorld::IdTable::headerData (int section,
|
|
|
|
Qt::Orientation orientation,
|
|
|
|
int role) const
|
2012-11-26 12:29:22 +01:00
|
|
|
{
|
|
|
|
if (orientation==Qt::Vertical)
|
|
|
|
return QVariant();
|
|
|
|
|
2012-12-13 14:53:16 +01:00
|
|
|
if (role==Qt::DisplayRole)
|
2013-08-07 09:36:05 +02:00
|
|
|
return tr (mIdCollection->getColumn (section).getTitle().c_str());
|
2012-12-13 14:53:16 +01:00
|
|
|
|
|
|
|
if (role==ColumnBase::Role_Flags)
|
|
|
|
return mIdCollection->getColumn (section).mFlags;
|
|
|
|
|
2013-01-03 11:20:25 +01:00
|
|
|
if (role==ColumnBase::Role_Display)
|
|
|
|
return mIdCollection->getColumn (section).mDisplayType;
|
|
|
|
|
2012-12-13 14:53:16 +01:00
|
|
|
return QVariant();
|
2012-11-29 14:45:34 +01:00
|
|
|
}
|
|
|
|
|
2014-06-19 18:46:09 +02:00
|
|
|
QVariant CSMWorld::IdTable::nestedHeaderData(int section, int subSection, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
2014-07-16 13:13:22 +02:00
|
|
|
const NestColumn& parentColumn = dynamic_cast<const NestColumn&>(mIdCollection->getColumn(section));
|
2014-06-19 18:46:09 +02:00
|
|
|
|
|
|
|
if (orientation==Qt::Vertical)
|
|
|
|
return QVariant();
|
|
|
|
|
|
|
|
if (role==Qt::DisplayRole)
|
2014-07-16 13:13:22 +02:00
|
|
|
return tr(parentColumn.nestedColumn(subSection).getTitle().c_str());
|
2014-06-19 18:46:09 +02:00
|
|
|
|
|
|
|
if (role==ColumnBase::Role_Flags)
|
|
|
|
return mIdCollection->getColumn (section).mFlags;
|
|
|
|
|
|
|
|
if (role==ColumnBase::Role_Display)
|
2014-07-16 13:13:22 +02:00
|
|
|
return parentColumn.nestedColumn(subSection).mDisplayType;
|
2014-06-19 18:46:09 +02:00
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
2014-06-17 18:28:49 +02:00
|
|
|
bool CSMWorld::IdTable::setData (const QModelIndex &index, const QVariant &value, int role)
|
2012-11-29 14:45:34 +01:00
|
|
|
{
|
2014-07-07 12:07:29 +02:00
|
|
|
if (index.internalId() != 0)
|
2012-11-29 14:45:34 +01:00
|
|
|
{
|
2014-07-07 12:07:29 +02:00
|
|
|
if (mIdCollection->getColumn(parent(index).column()).isEditable() && role==Qt::EditRole)
|
2014-06-09 10:35:39 +02:00
|
|
|
{
|
|
|
|
const std::pair<int, int>& parentAdress(unfoldIndexAdress(index.internalId()));
|
2012-12-06 14:56:04 +01:00
|
|
|
|
2014-06-09 10:35:39 +02:00
|
|
|
mIdCollection->setNestedData(parentAdress.first, parentAdress.second, value, index.row(), index.column());
|
2014-07-07 12:07:29 +02:00
|
|
|
|
|
|
|
emit dataChanged (CSMWorld::IdTable::index (parentAdress.first, 0),
|
|
|
|
CSMWorld::IdTable::index (parentAdress.second, mIdCollection->getColumns()-1));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
{
|
|
|
|
return false;
|
2014-06-09 10:35:39 +02:00
|
|
|
}
|
2012-11-29 14:45:34 +01:00
|
|
|
}
|
2014-07-07 12:07:29 +02:00
|
|
|
|
|
|
|
if (mIdCollection->getColumn (index.column()).isEditable() && role==Qt::EditRole)
|
|
|
|
{
|
|
|
|
mIdCollection->setData (index.row(), index.column(), value);
|
|
|
|
|
|
|
|
emit dataChanged (CSMWorld::IdTable::index (index.row(), 0),
|
|
|
|
CSMWorld::IdTable::index (index.row(), mIdCollection->getColumns()-1));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-11-29 14:45:34 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::ItemFlags CSMWorld::IdTable::flags (const QModelIndex & index) const
|
|
|
|
{
|
|
|
|
Qt::ItemFlags flags = Qt::ItemIsSelectable | Qt::ItemIsEnabled;
|
|
|
|
|
2012-12-13 14:53:16 +01:00
|
|
|
if (mIdCollection->getColumn (index.column()).isUserEditable())
|
2012-11-29 14:45:34 +01:00
|
|
|
flags |= Qt::ItemIsEditable;
|
|
|
|
|
|
|
|
return flags;
|
2012-12-03 21:44:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CSMWorld::IdTable::removeRows (int row, int count, const QModelIndex& parent)
|
|
|
|
{
|
2014-07-02 13:29:25 +02:00
|
|
|
beginRemoveRows (parent, row, row+count-1);
|
|
|
|
|
2012-12-03 21:44:16 +01:00
|
|
|
if (parent.isValid())
|
2014-06-24 12:21:40 +02:00
|
|
|
{
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
{
|
2014-07-07 12:07:29 +02:00
|
|
|
mIdCollection->removeNestedRows(parent.row(), parent.column(), row+i);
|
2014-06-24 12:21:40 +02:00
|
|
|
}
|
2014-07-07 12:07:29 +02:00
|
|
|
} else
|
2014-07-02 13:29:25 +02:00
|
|
|
{
|
2012-12-03 21:44:16 +01:00
|
|
|
|
2014-07-02 13:29:25 +02:00
|
|
|
beginRemoveRows (parent, row, row+count-1);
|
2012-12-03 21:44:16 +01:00
|
|
|
|
2014-07-02 13:29:25 +02:00
|
|
|
mIdCollection->removeRows (row, count);
|
|
|
|
}
|
2012-12-03 21:44:16 +01:00
|
|
|
|
|
|
|
endRemoveRows();
|
|
|
|
|
2014-07-07 12:22:04 +02:00
|
|
|
emit dataChanged (CSMWorld::IdTable::index (parent.row(), 0),
|
|
|
|
CSMWorld::IdTable::index (parent.row(), mIdCollection->getColumns()-1));
|
|
|
|
|
2012-12-03 21:44:16 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-06-24 19:03:29 +02:00
|
|
|
void CSMWorld::IdTable::addNestedRow(const QModelIndex& parent, int position)
|
|
|
|
{
|
|
|
|
assert(parent.isValid());
|
|
|
|
|
2014-07-01 21:13:27 +02:00
|
|
|
int row = parent.row();
|
2014-07-02 13:13:03 +02:00
|
|
|
|
|
|
|
beginInsertRows(parent, position, position);
|
2014-07-01 21:13:27 +02:00
|
|
|
mIdCollection->addNestedRow(row, parent.column(), position);
|
|
|
|
|
2014-07-07 12:22:04 +02:00
|
|
|
endInsertRows();
|
|
|
|
|
2014-07-01 21:13:27 +02:00
|
|
|
emit dataChanged (CSMWorld::IdTable::index (row, 0),
|
|
|
|
CSMWorld::IdTable::index (row, mIdCollection->getColumns()-1));
|
2014-06-24 19:03:29 +02:00
|
|
|
}
|
|
|
|
|
2013-03-21 10:07:25 +01:00
|
|
|
QModelIndex CSMWorld::IdTable::index (int row, int column, const QModelIndex& parent) const
|
|
|
|
{
|
2014-06-02 20:41:37 +02:00
|
|
|
unsigned int encodedId = 0;
|
2013-03-21 10:07:25 +01:00
|
|
|
if (parent.isValid())
|
2014-06-02 20:41:37 +02:00
|
|
|
{
|
|
|
|
encodedId = this->foldIndexAdress(parent);
|
|
|
|
}
|
2013-03-21 10:07:25 +01:00
|
|
|
|
|
|
|
if (row<0 || row>=mIdCollection->getSize())
|
|
|
|
return QModelIndex();
|
|
|
|
|
|
|
|
if (column<0 || column>=mIdCollection->getColumns())
|
|
|
|
return QModelIndex();
|
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
return createIndex(row, column, encodedId);
|
2013-03-21 10:07:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex CSMWorld::IdTable::parent (const QModelIndex& index) const
|
|
|
|
{
|
2014-06-09 10:35:39 +02:00
|
|
|
if (index.internalId() == 0) //0 is used for indexs with invalid parent (top level data)
|
|
|
|
{
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
unsigned int id = index.internalId();
|
|
|
|
const std::pair<int, int>& adress(unfoldIndexAdress(id));
|
2014-06-09 10:35:39 +02:00
|
|
|
|
|
|
|
if (adress.first >= this->rowCount() || adress.second >= this->columnCount())
|
|
|
|
{
|
|
|
|
throw "Parent index is not present in the model";
|
|
|
|
}
|
2014-06-02 20:41:37 +02:00
|
|
|
return createIndex(adress.first, adress.second);
|
2013-03-21 10:07:25 +01:00
|
|
|
}
|
|
|
|
|
2013-07-30 10:27:17 +02:00
|
|
|
void CSMWorld::IdTable::addRecord (const std::string& id, UniversalId::Type type)
|
2012-12-03 21:44:16 +01:00
|
|
|
{
|
2013-10-22 11:21:12 +02:00
|
|
|
int index = mIdCollection->getAppendIndex (id, type);
|
2012-12-03 21:44:16 +01:00
|
|
|
|
|
|
|
beginInsertRows (QModelIndex(), index, index);
|
|
|
|
|
2013-07-30 10:27:17 +02:00
|
|
|
mIdCollection->appendBlankRecord (id, type);
|
2012-12-03 21:44:16 +01:00
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
2014-01-20 13:59:00 +01:00
|
|
|
void CSMWorld::IdTable::cloneRecord(const std::string& origin,
|
2014-01-19 16:49:39 +01:00
|
|
|
const std::string& destination,
|
|
|
|
CSMWorld::UniversalId::Type type)
|
|
|
|
{
|
2014-01-20 18:28:06 +01:00
|
|
|
int index = mIdCollection->getAppendIndex (destination);
|
2014-05-25 15:46:23 +02:00
|
|
|
|
2014-01-20 18:28:06 +01:00
|
|
|
beginInsertRows (QModelIndex(), index, index);
|
2014-01-27 13:08:14 +01:00
|
|
|
mIdCollection->cloneRecord(origin, destination, type);
|
2014-01-20 18:28:06 +01:00
|
|
|
endInsertRows();
|
2014-01-19 16:49:39 +01:00
|
|
|
}
|
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
///This method can return only indexes to the top level table cells
|
2012-12-03 21:44:16 +01:00
|
|
|
QModelIndex CSMWorld::IdTable::getModelIndex (const std::string& id, int column) const
|
|
|
|
{
|
2014-06-02 20:41:37 +02:00
|
|
|
return index(mIdCollection->getIndex (id), column);
|
2012-12-06 14:56:04 +01:00
|
|
|
}
|
|
|
|
|
2013-04-29 17:52:01 +02:00
|
|
|
void CSMWorld::IdTable::setRecord (const std::string& id, const RecordBase& record)
|
2012-12-06 14:56:04 +01:00
|
|
|
{
|
2013-04-29 17:52:01 +02:00
|
|
|
int index = mIdCollection->searchId (id);
|
2012-12-06 14:56:04 +01:00
|
|
|
|
|
|
|
if (index==-1)
|
|
|
|
{
|
2013-10-22 11:21:12 +02:00
|
|
|
int index = mIdCollection->getAppendIndex (id);
|
2012-12-06 14:56:04 +01:00
|
|
|
|
|
|
|
beginInsertRows (QModelIndex(), index, index);
|
|
|
|
|
|
|
|
mIdCollection->appendRecord (record);
|
|
|
|
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mIdCollection->replace (index, record);
|
|
|
|
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 12:49:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::searchColumnIndex (Columns::ColumnId id) const
|
|
|
|
{
|
2013-09-27 15:04:09 +02:00
|
|
|
return mIdCollection->searchColumnIndex (id);
|
2013-08-08 12:49:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMWorld::IdTable::findColumnIndex (Columns::ColumnId id) const
|
|
|
|
{
|
2013-09-27 15:04:09 +02:00
|
|
|
return mIdCollection->findColumnIndex (id);
|
2013-11-14 11:39:14 +01: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),
|
|
|
|
index (baseIndex+newOrder.size()-1, mIdCollection->getColumns()-1));
|
|
|
|
}
|
|
|
|
|
2014-03-02 22:43:44 +01:00
|
|
|
std::pair<CSMWorld::UniversalId, std::string> CSMWorld::IdTable::view (int row) const
|
|
|
|
{
|
|
|
|
std::string id;
|
|
|
|
std::string hint;
|
|
|
|
|
2014-07-04 13:24:35 +02:00
|
|
|
if (getFeatures() & Feature_ViewCell)
|
2014-03-02 22:43:44 +01: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 13:24:35 +02:00
|
|
|
else if (getFeatures() & Feature_ViewId)
|
2014-03-02 22:43:44 +01: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]=='#')
|
|
|
|
id = "sys::default";
|
|
|
|
|
|
|
|
return std::make_pair (UniversalId (UniversalId::Type_Scene, id), hint);
|
2014-03-08 15:27:43 +01:00
|
|
|
}
|
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
///For top level data/columns
|
2014-07-04 13:24:35 +02:00
|
|
|
bool CSMWorld::IdTable::isDeleted (const std::string& id) const
|
|
|
|
{
|
|
|
|
return getRecord (id).isDeleted();
|
|
|
|
}
|
|
|
|
|
2014-03-08 15:27:43 +01:00
|
|
|
int CSMWorld::IdTable::getColumnId(int column) const
|
|
|
|
{
|
|
|
|
return mIdCollection->getColumn(column).getId();
|
2014-06-09 10:35:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int CSMWorld::IdTable::foldIndexAdress (const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
unsigned int out = index.row() * this->columnCount();
|
|
|
|
out += index.column();
|
2014-06-02 20:41:37 +02:00
|
|
|
return ++out;
|
2014-06-09 10:35:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::pair< int, int > CSMWorld::IdTable::unfoldIndexAdress (unsigned int id) const
|
|
|
|
{
|
|
|
|
if (id == 0)
|
|
|
|
{
|
|
|
|
throw "Attempt to unfold index id of the top level data cell";
|
|
|
|
}
|
|
|
|
|
|
|
|
--id;
|
|
|
|
int row = id / this->columnCount();
|
|
|
|
int column = id - row * this->columnCount();
|
|
|
|
return std::make_pair<int, int>(row, column);
|
|
|
|
}
|
2014-06-17 11:49:35 +02:00
|
|
|
|
|
|
|
bool CSMWorld::IdTable::hasChildren(const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
return (index.isValid() &&
|
|
|
|
index.internalId() == 0 &&
|
2014-07-16 13:13:22 +02:00
|
|
|
mIdCollection->getColumn(index.column()).mCanNest &&
|
2014-06-17 11:49:35 +02:00
|
|
|
index.data().isValid());
|
|
|
|
}
|
2014-07-18 18:26:22 +02:00
|
|
|
|
|
|
|
void CSMWorld::IdTable::setNestedTable(const QModelIndex& index, const CSMWorld::NestedTableWrapperBase& nestedTable)
|
|
|
|
{
|
|
|
|
if (!hasChildren(index))
|
|
|
|
{
|
|
|
|
throw std::logic_error("Tried to set nested table, but index has no children");
|
|
|
|
}
|
|
|
|
|
|
|
|
mIdCollection->setNestedTable(index.row(), index.column(), nestedTable);
|
2014-07-20 18:52:35 +02:00
|
|
|
|
|
|
|
emit dataChanged (CSMWorld::IdTable::index (index.row(), 0),
|
|
|
|
CSMWorld::IdTable::index (index.row(), mIdCollection->getColumns()-1));
|
2014-07-18 18:26:22 +02:00
|
|
|
}
|
|
|
|
|
2014-07-20 18:52:35 +02:00
|
|
|
CSMWorld::NestedTableWrapperBase* CSMWorld::IdTable::nestedTable(const QModelIndex& index) const
|
2014-07-18 18:26:22 +02:00
|
|
|
{
|
|
|
|
if (!hasChildren(index))
|
|
|
|
{
|
|
|
|
throw std::logic_error("Tried to retrive nested table, but index has no children");
|
|
|
|
}
|
|
|
|
|
|
|
|
return mIdCollection->nestedTable(index.row(), index.column());
|
|
|
|
}
|