2012-12-13 13:35:08 +01:00
|
|
|
#include "dialoguesubview.hpp"
|
|
|
|
|
2014-03-07 17:15:43 +01:00
|
|
|
#include <utility>
|
|
|
|
#include <memory>
|
2015-03-12 13:10:25 +11:00
|
|
|
#include <stdexcept>
|
2014-03-07 17:15:43 +01:00
|
|
|
|
2012-12-30 14:01:52 +01:00
|
|
|
#include <QGridLayout>
|
|
|
|
#include <QLabel>
|
2014-03-07 17:15:43 +01:00
|
|
|
#include <QSize>
|
2013-03-21 10:07:25 +01:00
|
|
|
#include <QAbstractItemModel>
|
2013-01-08 14:59:40 +01:00
|
|
|
#include <QDoubleSpinBox>
|
|
|
|
#include <QSpinBox>
|
|
|
|
#include <QLineEdit>
|
2014-03-07 17:15:43 +01:00
|
|
|
#include <QEvent>
|
2013-01-08 14:59:40 +01:00
|
|
|
#include <QDataWidgetMapper>
|
2014-03-10 14:11:49 +01:00
|
|
|
#include <QCheckBox>
|
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QPlainTextEdit>
|
|
|
|
#include <QComboBox>
|
2015-03-30 11:19:37 +11:00
|
|
|
#include <QHeaderView>
|
2015-05-18 22:08:36 +10:00
|
|
|
#include <QScrollBar>
|
2015-07-03 15:37:10 +03:00
|
|
|
#include <QMenu>
|
2012-12-30 14:01:52 +01:00
|
|
|
|
2015-03-30 16:41:55 +11:00
|
|
|
#include "../../model/world/nestedtableproxymodel.hpp"
|
2012-12-30 14:01:52 +01:00
|
|
|
#include "../../model/world/columnbase.hpp"
|
2013-01-19 14:29:14 +01:00
|
|
|
#include "../../model/world/idtable.hpp"
|
2015-04-02 20:19:15 +11:00
|
|
|
#include "../../model/world/idtree.hpp"
|
2014-03-08 15:27:43 +01:00
|
|
|
#include "../../model/world/columns.hpp"
|
2014-03-12 15:46:27 +01:00
|
|
|
#include "../../model/world/record.hpp"
|
2014-03-12 20:34:55 +01:00
|
|
|
#include "../../model/world/tablemimedata.hpp"
|
2015-04-17 01:27:36 +10:00
|
|
|
#include "../../model/world/idtree.hpp"
|
2014-03-17 19:18:18 +01:00
|
|
|
#include "../../model/world/commands.hpp"
|
2015-05-01 12:14:09 +10:00
|
|
|
#include "../../model/doc/document.hpp"
|
2015-12-15 10:40:00 +01:00
|
|
|
|
|
|
|
#include "../../model/prefs/state.hpp"
|
2012-12-30 14:01:52 +01:00
|
|
|
|
2015-06-12 18:33:55 +03:00
|
|
|
#include "../widget/coloreditor.hpp"
|
2015-06-20 17:33:36 +03:00
|
|
|
#include "../widget/droplineedit.hpp"
|
2015-06-12 18:33:55 +03:00
|
|
|
|
2014-03-06 20:10:13 +01:00
|
|
|
#include "recordstatusdelegate.hpp"
|
|
|
|
#include "util.hpp"
|
2014-03-18 09:36:22 +01:00
|
|
|
#include "tablebottombox.hpp"
|
2014-06-30 13:09:10 +02:00
|
|
|
#include "nestedtable.hpp"
|
2015-06-27 12:49:56 +02:00
|
|
|
#include "recordbuttonbar.hpp"
|
2014-03-11 09:14:13 +01:00
|
|
|
/*
|
|
|
|
==============================NotEditableSubDelegate==========================================
|
|
|
|
*/
|
|
|
|
CSVWorld::NotEditableSubDelegate::NotEditableSubDelegate(const CSMWorld::IdTable* table, QObject * parent) :
|
|
|
|
QAbstractItemDelegate(parent),
|
|
|
|
mTable(table)
|
|
|
|
{}
|
|
|
|
|
2015-03-14 12:42:46 +11:00
|
|
|
void CSVWorld::NotEditableSubDelegate::setEditorData (QWidget* editor, const QModelIndex& index) const
|
2014-03-11 09:14:13 +01:00
|
|
|
{
|
2015-03-14 12:42:46 +11:00
|
|
|
QLabel* label = qobject_cast<QLabel*>(editor);
|
|
|
|
if(!label)
|
|
|
|
return;
|
|
|
|
|
2014-03-11 09:14:13 +01:00
|
|
|
QVariant v = index.data(Qt::EditRole);
|
|
|
|
if (!v.isValid())
|
|
|
|
{
|
|
|
|
v = index.data(Qt::DisplayRole);
|
|
|
|
if (!v.isValid())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-26 15:50:36 +02:00
|
|
|
CSMWorld::Columns::ColumnId columnId = static_cast<CSMWorld::Columns::ColumnId> (
|
|
|
|
mTable->getColumnId (index.column()));
|
2015-07-13 12:52:18 +02:00
|
|
|
|
2014-03-11 09:14:13 +01:00
|
|
|
if (QVariant::String == v.type())
|
|
|
|
{
|
2015-03-14 12:42:46 +11:00
|
|
|
label->setText(v.toString());
|
|
|
|
}
|
2015-06-26 15:50:36 +02:00
|
|
|
else if (CSMWorld::Columns::hasEnums (columnId))
|
2014-03-11 09:14:13 +01:00
|
|
|
{
|
|
|
|
int data = v.toInt();
|
2015-06-26 15:50:36 +02:00
|
|
|
std::vector<std::string> enumNames (CSMWorld::Columns::getEnums (columnId));
|
2015-07-13 12:52:18 +02:00
|
|
|
|
2015-03-14 12:42:46 +11:00
|
|
|
label->setText(QString::fromUtf8(enumNames.at(data).c_str()));
|
2014-03-11 09:14:13 +01:00
|
|
|
}
|
2015-06-26 15:50:36 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
label->setText (v.toString());
|
|
|
|
}
|
2014-03-11 09:14:13 +01:00
|
|
|
}
|
|
|
|
|
2015-03-14 12:42:46 +11:00
|
|
|
void CSVWorld::NotEditableSubDelegate::setModelData(QWidget* editor, QAbstractItemModel* model, const QModelIndex& index) const
|
2014-03-11 09:14:13 +01:00
|
|
|
{
|
|
|
|
//not editable widgets will not save model data
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::NotEditableSubDelegate::paint (QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
//does nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
QSize CSVWorld::NotEditableSubDelegate::sizeHint (const QStyleOptionViewItem& option, const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
return QSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
QWidget* CSVWorld::NotEditableSubDelegate::createEditor (QWidget *parent,
|
|
|
|
const QStyleOptionViewItem& option,
|
2015-03-14 12:42:46 +11:00
|
|
|
const QModelIndex& index) const
|
2014-03-11 09:14:13 +01:00
|
|
|
{
|
2015-08-06 08:45:38 +02:00
|
|
|
QLabel *label = new QLabel(parent);
|
|
|
|
label->setTextInteractionFlags (Qt::TextSelectableByMouse);
|
|
|
|
return label;
|
2014-03-11 09:14:13 +01:00
|
|
|
}
|
2014-03-06 20:10:13 +01:00
|
|
|
|
2014-03-07 22:17:40 +01:00
|
|
|
/*
|
|
|
|
==============================DialogueDelegateDispatcherProxy==========================================
|
|
|
|
*/
|
2014-03-08 15:10:55 +01:00
|
|
|
CSVWorld::DialogueDelegateDispatcherProxy::refWrapper::refWrapper(const QModelIndex& index) :
|
2014-03-07 22:17:40 +01:00
|
|
|
mIndex(index)
|
|
|
|
{}
|
|
|
|
|
|
|
|
CSVWorld::DialogueDelegateDispatcherProxy::DialogueDelegateDispatcherProxy(QWidget* editor, CSMWorld::ColumnBase::Display display) :
|
|
|
|
mEditor(editor),
|
|
|
|
mDisplay(display),
|
|
|
|
mIndexWrapper(NULL)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::DialogueDelegateDispatcherProxy::editorDataCommited()
|
|
|
|
{
|
2014-03-08 16:50:42 +01:00
|
|
|
if (mIndexWrapper.get())
|
|
|
|
{
|
|
|
|
emit editorDataCommited(mEditor, mIndexWrapper->mIndex, mDisplay);
|
|
|
|
}
|
2014-03-07 22:17:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::DialogueDelegateDispatcherProxy::setIndex(const QModelIndex& index)
|
|
|
|
{
|
|
|
|
mIndexWrapper.reset(new refWrapper(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
QWidget* CSVWorld::DialogueDelegateDispatcherProxy::getEditor() const
|
|
|
|
{
|
|
|
|
return mEditor;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
==============================DialogueDelegateDispatcher==========================================
|
|
|
|
*/
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
CSVWorld::DialogueDelegateDispatcher::DialogueDelegateDispatcher(QObject* parent,
|
2015-04-25 09:39:37 +10:00
|
|
|
CSMWorld::IdTable* table, CSMWorld::CommandDispatcher& commandDispatcher,
|
|
|
|
CSMDoc::Document& document, QAbstractItemModel *model) :
|
2014-03-07 17:15:43 +01:00
|
|
|
mParent(parent),
|
2015-04-17 01:27:36 +10:00
|
|
|
mTable(model ? model : table),
|
2015-04-25 09:39:37 +10:00
|
|
|
mCommandDispatcher (commandDispatcher), mDocument (document),
|
2014-03-11 09:14:13 +01:00
|
|
|
mNotEditableDelegate(table, parent)
|
2014-03-07 22:17:40 +01:00
|
|
|
{
|
|
|
|
}
|
2014-03-07 17:15:43 +01:00
|
|
|
|
|
|
|
CSVWorld::CommandDelegate* CSVWorld::DialogueDelegateDispatcher::makeDelegate(CSMWorld::ColumnBase::Display display)
|
|
|
|
{
|
|
|
|
CommandDelegate *delegate = NULL;
|
|
|
|
std::map<int, CommandDelegate*>::const_iterator delegateIt(mDelegates.find(display));
|
|
|
|
if (delegateIt == mDelegates.end())
|
|
|
|
{
|
|
|
|
delegate = CommandDelegateFactoryCollection::get().makeDelegate (
|
2015-01-15 14:24:33 +01:00
|
|
|
display, &mCommandDispatcher, mDocument, mParent);
|
2014-04-29 22:46:55 +10:00
|
|
|
mDelegates.insert(std::make_pair(display, delegate));
|
2014-03-07 17:15:43 +01:00
|
|
|
} else
|
|
|
|
{
|
|
|
|
delegate = delegateIt->second;
|
|
|
|
}
|
|
|
|
return delegate;
|
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
void CSVWorld::DialogueDelegateDispatcher::editorDataCommited(QWidget* editor,
|
|
|
|
const QModelIndex& index, CSMWorld::ColumnBase::Display display)
|
2014-03-07 17:15:43 +01:00
|
|
|
{
|
2014-03-07 22:17:40 +01:00
|
|
|
setModelData(editor, mTable, index, display);
|
2014-03-07 17:15:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::DialogueDelegateDispatcher::setEditorData (QWidget* editor, const QModelIndex& index) const
|
|
|
|
{
|
2015-05-29 05:40:20 +10:00
|
|
|
CSMWorld::ColumnBase::Display display = CSMWorld::ColumnBase::Display_None;
|
|
|
|
if (index.parent().isValid())
|
|
|
|
{
|
|
|
|
display = static_cast<CSMWorld::ColumnBase::Display>
|
|
|
|
(static_cast<CSMWorld::IdTree *>(mTable)->nestedHeaderData (index.parent().column(), index.column(), Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
display = static_cast<CSMWorld::ColumnBase::Display>
|
|
|
|
(mTable->headerData (index.column(), Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
|
|
|
}
|
2014-03-07 17:15:43 +01:00
|
|
|
|
2014-03-10 17:52:45 +01:00
|
|
|
QLabel* label = qobject_cast<QLabel*>(editor);
|
|
|
|
if(label)
|
|
|
|
{
|
2014-03-11 09:14:13 +01:00
|
|
|
mNotEditableDelegate.setEditorData(label, index);
|
2014-03-10 17:52:45 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-03-07 17:15:43 +01:00
|
|
|
std::map<int, CommandDelegate*>::const_iterator delegateIt(mDelegates.find(display));
|
|
|
|
if (delegateIt != mDelegates.end())
|
|
|
|
{
|
2014-03-10 09:47:41 +01:00
|
|
|
delegateIt->second->setEditorData(editor, index, true);
|
2014-03-07 17:15:43 +01:00
|
|
|
}
|
2014-03-07 22:17:40 +01:00
|
|
|
|
|
|
|
for (unsigned i = 0; i < mProxys.size(); ++i)
|
|
|
|
{
|
|
|
|
if (mProxys[i]->getEditor() == editor)
|
|
|
|
{
|
|
|
|
mProxys[i]->setIndex(index);
|
|
|
|
}
|
|
|
|
}
|
2014-03-07 17:15:43 +01:00
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
void CSVWorld::DialogueDelegateDispatcher::setModelData(QWidget* editor,
|
|
|
|
QAbstractItemModel* model, const QModelIndex& index) const
|
2015-03-14 12:42:46 +11:00
|
|
|
{
|
|
|
|
setModelData(editor, model, index, CSMWorld::ColumnBase::Display_None);
|
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
void CSVWorld::DialogueDelegateDispatcher::setModelData(QWidget* editor,
|
|
|
|
QAbstractItemModel* model, const QModelIndex& index, CSMWorld::ColumnBase::Display display) const
|
2014-03-07 17:15:43 +01:00
|
|
|
{
|
|
|
|
std::map<int, CommandDelegate*>::const_iterator delegateIt(mDelegates.find(display));
|
|
|
|
if (delegateIt != mDelegates.end())
|
|
|
|
{
|
|
|
|
delegateIt->second->setModelData(editor, model, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
void CSVWorld::DialogueDelegateDispatcher::paint (QPainter* painter,
|
|
|
|
const QStyleOptionViewItem& option, const QModelIndex& index) const
|
2014-03-07 17:15:43 +01:00
|
|
|
{
|
|
|
|
//Does nothing
|
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
QSize CSVWorld::DialogueDelegateDispatcher::sizeHint (const QStyleOptionViewItem& option,
|
|
|
|
const QModelIndex& index) const
|
2014-03-07 17:15:43 +01:00
|
|
|
{
|
|
|
|
return QSize(); //silencing warning, otherwise does nothing
|
|
|
|
}
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
QWidget* CSVWorld::DialogueDelegateDispatcher::makeEditor(CSMWorld::ColumnBase::Display display,
|
|
|
|
const QModelIndex& index)
|
2014-03-07 17:15:43 +01:00
|
|
|
{
|
2014-03-10 13:25:01 +01:00
|
|
|
QVariant variant = index.data();
|
|
|
|
if (!variant.isValid())
|
2014-03-08 16:50:42 +01:00
|
|
|
{
|
2014-03-10 13:25:01 +01:00
|
|
|
variant = index.data(Qt::DisplayRole);
|
|
|
|
if (!variant.isValid())
|
|
|
|
{
|
2014-03-10 14:11:49 +01:00
|
|
|
return NULL;
|
2014-03-10 13:25:01 +01:00
|
|
|
}
|
2014-03-08 16:50:42 +01:00
|
|
|
}
|
|
|
|
|
2014-03-07 17:15:43 +01:00
|
|
|
QWidget* editor = NULL;
|
2014-03-10 17:52:45 +01:00
|
|
|
if (! (mTable->flags (index) & Qt::ItemIsEditable))
|
|
|
|
{
|
2015-04-17 01:27:36 +10:00
|
|
|
return mNotEditableDelegate.createEditor(qobject_cast<QWidget*>(mParent),
|
|
|
|
QStyleOptionViewItem(), index);
|
2014-03-10 17:52:45 +01:00
|
|
|
}
|
|
|
|
|
2014-03-07 17:15:43 +01:00
|
|
|
std::map<int, CommandDelegate*>::iterator delegateIt(mDelegates.find(display));
|
2014-06-09 10:26:53 +02:00
|
|
|
|
2014-03-07 17:15:43 +01:00
|
|
|
if (delegateIt != mDelegates.end())
|
|
|
|
{
|
2015-04-17 01:27:36 +10:00
|
|
|
editor = delegateIt->second->createEditor(qobject_cast<QWidget*>(mParent),
|
|
|
|
QStyleOptionViewItem(), index, display);
|
2014-06-09 13:31:15 +02:00
|
|
|
|
2014-03-07 22:17:40 +01:00
|
|
|
DialogueDelegateDispatcherProxy* proxy = new DialogueDelegateDispatcherProxy(editor, display);
|
2014-03-10 14:11:49 +01:00
|
|
|
|
2015-03-12 13:10:25 +11:00
|
|
|
// NOTE: For each entry in CSVWorld::CommandDelegate::createEditor() a corresponding entry
|
|
|
|
// is required here
|
2015-06-20 17:33:36 +03:00
|
|
|
if (qobject_cast<CSVWidget::DropLineEdit*>(editor))
|
2014-03-10 14:11:49 +01:00
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(editingFinished()), proxy, SLOT(editorDataCommited()));
|
2014-06-09 13:31:15 +02:00
|
|
|
|
2015-06-20 20:52:47 +03:00
|
|
|
connect(editor, SIGNAL(tableMimeDataDropped(const CSMWorld::UniversalId&, const CSMDoc::Document*)),
|
|
|
|
proxy, SLOT(editorDataCommited()));
|
2014-03-10 14:11:49 +01:00
|
|
|
}
|
2014-09-01 11:15:59 +02:00
|
|
|
else if (qobject_cast<QCheckBox*>(editor))
|
2014-03-10 14:11:49 +01:00
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(stateChanged(int)), proxy, SLOT(editorDataCommited()));
|
|
|
|
}
|
2014-09-01 11:15:59 +02:00
|
|
|
else if (qobject_cast<QPlainTextEdit*>(editor))
|
2014-03-10 14:11:49 +01:00
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(textChanged()), proxy, SLOT(editorDataCommited()));
|
|
|
|
}
|
2014-09-01 11:15:59 +02:00
|
|
|
else if (qobject_cast<QComboBox*>(editor))
|
2014-03-10 14:11:49 +01:00
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(currentIndexChanged (int)), proxy, SLOT(editorDataCommited()));
|
|
|
|
}
|
2015-03-12 13:10:25 +11:00
|
|
|
else if (qobject_cast<QAbstractSpinBox*>(editor) || qobject_cast<QLineEdit*>(editor))
|
2014-03-12 20:34:55 +01:00
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(editingFinished()), proxy, SLOT(editorDataCommited()));
|
|
|
|
}
|
2015-06-12 18:33:55 +03:00
|
|
|
else if (qobject_cast<CSVWidget::ColorEditor *>(editor))
|
|
|
|
{
|
|
|
|
connect(editor, SIGNAL(pickingFinished()), proxy, SLOT(editorDataCommited()));
|
|
|
|
}
|
2015-03-12 13:10:25 +11:00
|
|
|
else // throw an exception because this is a coding error
|
|
|
|
throw std::logic_error ("Dialogue editor type missing");
|
2014-03-10 14:11:49 +01:00
|
|
|
|
2014-06-09 13:31:15 +02:00
|
|
|
connect(proxy, SIGNAL(editorDataCommited(QWidget*, const QModelIndex&, CSMWorld::ColumnBase::Display)),
|
2014-06-17 18:28:49 +02:00
|
|
|
this, SLOT(editorDataCommited(QWidget*, const QModelIndex&, CSMWorld::ColumnBase::Display)));
|
2014-03-10 14:11:49 +01:00
|
|
|
|
2014-03-07 22:17:40 +01:00
|
|
|
mProxys.push_back(proxy); //deleted in the destructor
|
2014-03-07 17:15:43 +01:00
|
|
|
}
|
|
|
|
return editor;
|
|
|
|
}
|
|
|
|
|
2014-03-07 22:17:40 +01:00
|
|
|
CSVWorld::DialogueDelegateDispatcher::~DialogueDelegateDispatcher()
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < mProxys.size(); ++i)
|
|
|
|
{
|
|
|
|
delete mProxys[i]; //unique_ptr could be handy
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:37:10 +03:00
|
|
|
|
|
|
|
CSVWorld::IdContextMenu::IdContextMenu(QWidget *widget, CSMWorld::ColumnBase::Display display)
|
|
|
|
: QObject(widget),
|
|
|
|
mWidget(widget),
|
|
|
|
mIdType(CSMWorld::TableMimeData::convertEnums(display))
|
|
|
|
{
|
|
|
|
Q_ASSERT(mWidget != NULL);
|
|
|
|
Q_ASSERT(CSMWorld::ColumnBase::isId(display));
|
|
|
|
Q_ASSERT(mIdType != CSMWorld::UniversalId::Type_None);
|
2015-07-13 12:52:18 +02:00
|
|
|
|
2015-07-03 15:37:10 +03:00
|
|
|
mWidget->setContextMenuPolicy(Qt::CustomContextMenu);
|
2015-07-13 12:52:18 +02:00
|
|
|
connect(mWidget,
|
|
|
|
SIGNAL(customContextMenuRequested(const QPoint &)),
|
|
|
|
this,
|
2015-07-03 15:37:10 +03:00
|
|
|
SLOT(showContextMenu(const QPoint &)));
|
|
|
|
|
|
|
|
mEditIdAction = new QAction(this);
|
2015-07-03 16:24:08 +03:00
|
|
|
connect(mEditIdAction, SIGNAL(triggered()), this, SLOT(editIdRequest()));
|
2015-07-03 15:37:10 +03:00
|
|
|
|
|
|
|
QLineEdit *lineEdit = qobject_cast<QLineEdit *>(mWidget);
|
|
|
|
if (lineEdit != NULL)
|
|
|
|
{
|
|
|
|
mContextMenu = lineEdit->createStandardContextMenu();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mContextMenu = new QMenu(mWidget);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 18:48:13 +03:00
|
|
|
void CSVWorld::IdContextMenu::excludeId(const std::string &id)
|
|
|
|
{
|
|
|
|
mExcludedIds.insert(id);
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:37:10 +03:00
|
|
|
QString CSVWorld::IdContextMenu::getWidgetValue() const
|
|
|
|
{
|
2015-07-13 12:52:18 +02:00
|
|
|
QLineEdit *lineEdit = qobject_cast<QLineEdit *>(mWidget);
|
2015-07-03 16:24:08 +03:00
|
|
|
QLabel *label = qobject_cast<QLabel *>(mWidget);
|
2015-07-03 15:37:10 +03:00
|
|
|
|
|
|
|
QString value = "";
|
|
|
|
if (lineEdit != NULL)
|
|
|
|
{
|
|
|
|
value = lineEdit->text();
|
|
|
|
}
|
|
|
|
else if (label != NULL)
|
|
|
|
{
|
|
|
|
value = label->text();
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2015-07-03 16:51:47 +03:00
|
|
|
void CSVWorld::IdContextMenu::addEditIdActionToMenu(const QString &text)
|
|
|
|
{
|
|
|
|
mEditIdAction->setText(text);
|
|
|
|
if (mContextMenu->actions().isEmpty())
|
|
|
|
{
|
|
|
|
mContextMenu->addAction(mEditIdAction);
|
|
|
|
}
|
2015-07-03 19:07:44 +03:00
|
|
|
else if (mContextMenu->actions().first() != mEditIdAction)
|
2015-07-03 16:51:47 +03:00
|
|
|
{
|
|
|
|
QAction *action = mContextMenu->actions().first();
|
|
|
|
mContextMenu->insertAction(action, mEditIdAction);
|
|
|
|
mContextMenu->insertSeparator(action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::IdContextMenu::removeEditIdActionFromMenu()
|
|
|
|
{
|
|
|
|
if (mContextMenu->actions().isEmpty())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mContextMenu->actions().first() == mEditIdAction)
|
|
|
|
{
|
|
|
|
mContextMenu->removeAction(mEditIdAction);
|
|
|
|
if (!mContextMenu->actions().isEmpty() && mContextMenu->actions().first()->isSeparator())
|
|
|
|
{
|
|
|
|
mContextMenu->removeAction(mContextMenu->actions().first());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:37:10 +03:00
|
|
|
void CSVWorld::IdContextMenu::showContextMenu(const QPoint &pos)
|
|
|
|
{
|
|
|
|
QString value = getWidgetValue();
|
2015-07-03 18:48:13 +03:00
|
|
|
bool isExcludedId = mExcludedIds.find(value.toUtf8().constData()) != mExcludedIds.end();
|
|
|
|
if (!value.isEmpty() && !isExcludedId)
|
2015-07-03 15:37:10 +03:00
|
|
|
{
|
2015-07-03 16:51:47 +03:00
|
|
|
addEditIdActionToMenu("Edit '" + value + "'");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
removeEditIdActionFromMenu();
|
|
|
|
}
|
2015-07-13 12:52:18 +02:00
|
|
|
|
2015-07-03 16:51:47 +03:00
|
|
|
if (!mContextMenu->actions().isEmpty())
|
|
|
|
{
|
2015-07-03 16:24:08 +03:00
|
|
|
mContextMenu->exec(mWidget->mapToGlobal(pos));
|
2015-07-03 15:37:10 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 16:24:08 +03:00
|
|
|
void CSVWorld::IdContextMenu::editIdRequest()
|
|
|
|
{
|
|
|
|
CSMWorld::UniversalId editId(mIdType, getWidgetValue().toUtf8().constData());
|
|
|
|
emit editIdRequest(editId, "");
|
|
|
|
}
|
|
|
|
|
2014-03-07 22:17:40 +01:00
|
|
|
/*
|
2014-03-12 10:21:52 +01:00
|
|
|
=============================================================EditWidget=====================================================
|
2014-03-07 22:17:40 +01:00
|
|
|
*/
|
|
|
|
|
2015-07-04 19:55:48 +03:00
|
|
|
void CSVWorld::EditWidget::createEditorContextMenu(QWidget *editor,
|
|
|
|
CSMWorld::ColumnBase::Display display,
|
|
|
|
int currentRow) const
|
|
|
|
{
|
|
|
|
Q_ASSERT(editor != NULL);
|
|
|
|
|
|
|
|
if (CSMWorld::ColumnBase::isId(display) &&
|
|
|
|
CSMWorld::TableMimeData::convertEnums(display) != CSMWorld::UniversalId::Type_None)
|
|
|
|
{
|
|
|
|
int idColumn = mTable->findColumnIndex(CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QString id = mTable->data(mTable->index(currentRow, idColumn)).toString();
|
|
|
|
|
|
|
|
IdContextMenu *menu = new IdContextMenu(editor, display);
|
|
|
|
// Current ID is already opened, so no need to create Edit 'ID' action for it
|
|
|
|
menu->excludeId(id.toUtf8().constData());
|
|
|
|
connect(menu,
|
|
|
|
SIGNAL(editIdRequest(const CSMWorld::UniversalId &, const std::string &)),
|
|
|
|
this,
|
|
|
|
SIGNAL(editIdRequest(const CSMWorld::UniversalId &, const std::string &)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-18 16:53:46 +02:00
|
|
|
CSVWorld::EditWidget::~EditWidget()
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < mNestedModels.size(); ++i)
|
|
|
|
delete mNestedModels[i];
|
2015-05-19 19:19:52 +10:00
|
|
|
|
|
|
|
if (mDispatcher)
|
|
|
|
delete mDispatcher;
|
|
|
|
|
|
|
|
if (mNestedTableDispatcher)
|
|
|
|
delete mNestedTableDispatcher;
|
2014-06-18 16:53:46 +02:00
|
|
|
}
|
|
|
|
|
2015-04-25 09:39:37 +10:00
|
|
|
CSVWorld::EditWidget::EditWidget(QWidget *parent,
|
|
|
|
int row, CSMWorld::IdTable* table, CSMWorld::CommandDispatcher& commandDispatcher,
|
|
|
|
CSMDoc::Document& document, bool createAndDelete) :
|
2014-03-12 10:21:52 +01:00
|
|
|
QScrollArea(parent),
|
|
|
|
mWidgetMapper(NULL),
|
2015-04-17 01:27:36 +10:00
|
|
|
mNestedTableMapper(NULL),
|
2015-05-24 19:24:27 +10:00
|
|
|
mDispatcher(NULL),
|
2015-04-30 19:24:27 -05:00
|
|
|
mNestedTableDispatcher(NULL),
|
2014-03-12 10:21:52 +01:00
|
|
|
mMainWidget(NULL),
|
2015-04-30 19:24:27 -05:00
|
|
|
mTable(table),
|
2015-01-15 14:24:33 +01:00
|
|
|
mCommandDispatcher (commandDispatcher),
|
2015-04-30 19:24:27 -05:00
|
|
|
mDocument (document)
|
2014-03-12 10:21:52 +01:00
|
|
|
{
|
2014-03-12 12:25:37 +01:00
|
|
|
remake (row);
|
2014-03-12 10:21:52 +01:00
|
|
|
}
|
2014-03-07 17:15:43 +01:00
|
|
|
|
2014-03-12 12:25:37 +01:00
|
|
|
void CSVWorld::EditWidget::remake(int row)
|
2012-12-13 13:35:08 +01:00
|
|
|
{
|
2015-05-25 22:28:41 +10:00
|
|
|
if (mMainWidget)
|
|
|
|
{
|
|
|
|
QWidget *del = this->takeWidget();
|
|
|
|
del->deleteLater();
|
|
|
|
}
|
|
|
|
mMainWidget = new QWidget (this);
|
|
|
|
|
2014-06-18 16:53:46 +02:00
|
|
|
for (unsigned i = 0; i < mNestedModels.size(); ++i)
|
|
|
|
delete mNestedModels[i];
|
2015-05-19 19:19:52 +10:00
|
|
|
|
2014-07-07 10:23:40 +02:00
|
|
|
mNestedModels.clear();
|
2015-03-06 14:36:13 +11:00
|
|
|
|
2015-05-19 19:19:52 +10:00
|
|
|
if (mDispatcher)
|
|
|
|
delete mDispatcher;
|
|
|
|
mDispatcher = new DialogueDelegateDispatcher(0/*this*/, mTable, mCommandDispatcher, mDocument);
|
|
|
|
|
|
|
|
if (mNestedTableDispatcher)
|
|
|
|
delete mNestedTableDispatcher;
|
2014-03-11 18:38:37 +01:00
|
|
|
|
2014-03-16 17:11:13 +01:00
|
|
|
//not sure if widget mapper can handle deleting the widgets that were mapped
|
2014-03-12 10:21:52 +01:00
|
|
|
if (mWidgetMapper)
|
|
|
|
delete mWidgetMapper;
|
2015-05-19 19:19:52 +10:00
|
|
|
|
|
|
|
mWidgetMapper = new QDataWidgetMapper (this);
|
|
|
|
mWidgetMapper->setModel(mTable);
|
|
|
|
mWidgetMapper->setItemDelegate(mDispatcher);
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
if (mNestedTableMapper)
|
|
|
|
delete mNestedTableMapper;
|
2014-06-09 13:31:15 +02:00
|
|
|
|
2012-12-30 14:01:52 +01:00
|
|
|
|
2014-03-12 10:21:52 +01:00
|
|
|
QFrame* line = new QFrame(mMainWidget);
|
2014-03-09 18:44:04 +01:00
|
|
|
line->setObjectName(QString::fromUtf8("line"));
|
|
|
|
line->setGeometry(QRect(320, 150, 118, 3));
|
|
|
|
line->setFrameShape(QFrame::HLine);
|
|
|
|
line->setFrameShadow(QFrame::Sunken);
|
|
|
|
|
2015-04-19 10:32:06 +10:00
|
|
|
QFrame* line2 = new QFrame(mMainWidget);
|
|
|
|
line2->setObjectName(QString::fromUtf8("line"));
|
|
|
|
line2->setGeometry(QRect(320, 150, 118, 3));
|
|
|
|
line2->setFrameShape(QFrame::HLine);
|
|
|
|
line2->setFrameShadow(QFrame::Sunken);
|
|
|
|
|
2014-03-12 10:21:52 +01:00
|
|
|
QVBoxLayout *mainLayout = new QVBoxLayout(mMainWidget);
|
|
|
|
QGridLayout *lockedLayout = new QGridLayout();
|
2015-04-09 19:39:09 +10:00
|
|
|
QGridLayout *unlockedLayout = new QGridLayout();
|
2014-06-17 18:28:49 +02:00
|
|
|
QVBoxLayout *tablesLayout = new QVBoxLayout();
|
|
|
|
|
2015-04-09 19:39:09 +10:00
|
|
|
mainLayout->addLayout(lockedLayout, QSizePolicy::Fixed);
|
2014-03-09 18:44:04 +01:00
|
|
|
mainLayout->addWidget(line, 1);
|
2015-04-19 10:32:06 +10:00
|
|
|
mainLayout->addLayout(unlockedLayout, QSizePolicy::Preferred);
|
|
|
|
mainLayout->addWidget(line2, 1);
|
2015-04-09 19:39:09 +10:00
|
|
|
mainLayout->addLayout(tablesLayout, QSizePolicy::Preferred);
|
2014-03-09 18:44:04 +01:00
|
|
|
mainLayout->addStretch(1);
|
|
|
|
|
|
|
|
int unlocked = 0;
|
|
|
|
int locked = 0;
|
2014-03-12 10:21:52 +01:00
|
|
|
const int columns = mTable->columnCount();
|
2014-06-02 20:41:37 +02:00
|
|
|
|
2012-12-30 14:01:52 +01:00
|
|
|
for (int i=0; i<columns; ++i)
|
|
|
|
{
|
2014-03-12 10:21:52 +01:00
|
|
|
int flags = mTable->headerData (i, Qt::Horizontal, CSMWorld::ColumnBase::Role_Flags).toInt();
|
2012-12-13 13:35:08 +01:00
|
|
|
|
2012-12-30 14:01:52 +01:00
|
|
|
if (flags & CSMWorld::ColumnBase::Flag_Dialogue)
|
|
|
|
{
|
2013-01-08 14:59:40 +01:00
|
|
|
CSMWorld::ColumnBase::Display display = static_cast<CSMWorld::ColumnBase::Display>
|
2014-03-12 10:21:52 +01:00
|
|
|
(mTable->headerData (i, Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt());
|
2013-01-08 14:59:40 +01:00
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
if (mTable->hasChildren(mTable->index(row, i)) &&
|
|
|
|
!(flags & CSMWorld::ColumnBase::Flag_Dialogue_List))
|
2013-01-08 14:59:40 +01:00
|
|
|
{
|
2015-05-01 12:14:09 +10:00
|
|
|
mNestedModels.push_back(new CSMWorld::NestedTableProxyModel (
|
|
|
|
mTable->index(row, i), display, dynamic_cast<CSMWorld::IdTree*>(mTable)));
|
2015-03-06 14:36:13 +11:00
|
|
|
|
2015-04-25 09:39:37 +10:00
|
|
|
int idColumn = mTable->findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
int typeColumn = mTable->findColumnIndex (CSMWorld::Columns::ColumnId_RecordType);
|
|
|
|
|
|
|
|
CSMWorld::UniversalId id = CSMWorld::UniversalId(
|
|
|
|
static_cast<CSMWorld::UniversalId::Type> (mTable->data (mTable->index (row, typeColumn)).toInt()),
|
|
|
|
mTable->data (mTable->index (row, idColumn)).toString().toUtf8().constData());
|
|
|
|
|
2015-10-28 20:30:30 +11:00
|
|
|
bool editable = true;
|
|
|
|
bool fixedRows = false;
|
|
|
|
QVariant v = mTable->index(row, i).data();
|
2015-10-28 23:33:24 +11:00
|
|
|
if (v.canConvert<CSMWorld::ColumnBase::TableEditModes>())
|
2015-10-28 20:30:30 +11:00
|
|
|
{
|
2015-10-28 23:33:24 +11:00
|
|
|
assert (QString(v.typeName()) == "CSMWorld::ColumnBase::TableEditModes");
|
2015-10-28 20:30:30 +11:00
|
|
|
|
2015-10-28 23:33:24 +11:00
|
|
|
if (v.value<CSMWorld::ColumnBase::TableEditModes>() == CSMWorld::ColumnBase::TableEdit_None)
|
2015-10-28 20:30:30 +11:00
|
|
|
editable = false;
|
2015-10-28 23:33:24 +11:00
|
|
|
else if (v.value<CSMWorld::ColumnBase::TableEditModes>() == CSMWorld::ColumnBase::TableEdit_FixedRows)
|
2015-10-28 20:30:30 +11:00
|
|
|
fixedRows = true;
|
|
|
|
}
|
2015-05-21 13:11:07 +10:00
|
|
|
|
2015-10-28 20:52:07 +11:00
|
|
|
NestedTable* table =
|
|
|
|
new NestedTable(mDocument, id, mNestedModels.back(), this, editable, fixedRows);
|
|
|
|
table->resizeColumnsToContents();
|
2015-10-28 20:30:30 +11:00
|
|
|
if (!editable)
|
2015-05-21 13:11:07 +10:00
|
|
|
{
|
|
|
|
table->setEditTriggers(QAbstractItemView::NoEditTriggers);
|
|
|
|
table->setEnabled(false);
|
|
|
|
}
|
2015-03-30 11:19:37 +11:00
|
|
|
|
2015-04-19 21:07:45 +10:00
|
|
|
int rows = mTable->rowCount(mTable->index(row, i));
|
2015-04-19 10:32:06 +10:00
|
|
|
int rowHeight = (rows == 0) ? table->horizontalHeader()->height() : table->rowHeight(0);
|
|
|
|
int tableMaxHeight = (5 * rowHeight)
|
|
|
|
+ table->horizontalHeader()->height() + 2 * table->frameWidth();
|
|
|
|
table->setMinimumHeight(tableMaxHeight);
|
|
|
|
|
2015-04-17 01:27:36 +10:00
|
|
|
QLabel* label =
|
|
|
|
new QLabel (mTable->headerData (i, Qt::Horizontal, Qt::DisplayRole).toString(), mMainWidget);
|
2014-06-18 16:53:46 +02:00
|
|
|
|
2015-03-30 11:19:37 +11:00
|
|
|
label->setSizePolicy (QSizePolicy::Fixed, QSizePolicy::Fixed);
|
2015-10-28 20:30:30 +11:00
|
|
|
if(!editable)
|
2015-05-21 13:11:07 +10:00
|
|
|
label->setEnabled(false);
|
2013-01-08 14:59:40 +01:00
|
|
|
|
2015-03-30 11:19:37 +11:00
|
|
|
tablesLayout->addWidget(label);
|
2014-06-17 18:28:49 +02:00
|
|
|
tablesLayout->addWidget(table);
|
2015-07-04 19:27:42 +03:00
|
|
|
|
2015-07-13 12:52:18 +02:00
|
|
|
connect(table,
|
|
|
|
SIGNAL(editRequest(const CSMWorld::UniversalId &, const std::string &)),
|
|
|
|
this,
|
2015-07-04 19:27:42 +03:00
|
|
|
SIGNAL(editIdRequest(const CSMWorld::UniversalId &, const std::string &)));
|
2015-03-30 11:19:37 +11:00
|
|
|
}
|
2015-04-17 01:27:36 +10:00
|
|
|
else if (!(flags & CSMWorld::ColumnBase::Flag_Dialogue_List))
|
2013-01-08 14:59:40 +01:00
|
|
|
{
|
2015-05-19 19:19:52 +10:00
|
|
|
mDispatcher->makeDelegate (display);
|
|
|
|
QWidget* editor = mDispatcher->makeEditor (display, (mTable->index (row, i)));
|
2014-06-02 20:41:37 +02:00
|
|
|
|
|
|
|
if (editor)
|
2014-03-08 15:18:40 +01:00
|
|
|
{
|
2014-06-02 20:41:37 +02:00
|
|
|
mWidgetMapper->addMapping (editor, i);
|
2014-06-17 11:49:35 +02:00
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
QLabel* label = new QLabel (mTable->headerData (i, Qt::Horizontal).toString(), mMainWidget);
|
2014-06-17 11:49:35 +02:00
|
|
|
|
2014-06-02 20:41:37 +02:00
|
|
|
label->setSizePolicy (QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
|
|
editor->setSizePolicy (QSizePolicy::MinimumExpanding, QSizePolicy::Fixed);
|
|
|
|
|
|
|
|
if (! (mTable->flags (mTable->index (row, i)) & Qt::ItemIsEditable))
|
|
|
|
{
|
|
|
|
lockedLayout->addWidget (label, locked, 0);
|
|
|
|
lockedLayout->addWidget (editor, locked, 1);
|
|
|
|
++locked;
|
2015-03-30 11:19:37 +11:00
|
|
|
}
|
|
|
|
else
|
2014-06-02 20:41:37 +02:00
|
|
|
{
|
|
|
|
unlockedLayout->addWidget (label, unlocked, 0);
|
|
|
|
unlockedLayout->addWidget (editor, unlocked, 1);
|
|
|
|
++unlocked;
|
|
|
|
}
|
2015-05-21 13:11:07 +10:00
|
|
|
|
|
|
|
if(mTable->index(row, i).data().type() == QVariant::UserType)
|
|
|
|
{
|
|
|
|
editor->setEnabled(false);
|
|
|
|
label->setEnabled(false);
|
|
|
|
}
|
2015-07-03 16:24:08 +03:00
|
|
|
|
2015-07-04 19:55:48 +03:00
|
|
|
createEditorContextMenu(editor, display, row);
|
2014-03-08 15:18:40 +01:00
|
|
|
}
|
2013-01-08 14:59:40 +01:00
|
|
|
}
|
2015-04-17 01:27:36 +10:00
|
|
|
else
|
|
|
|
{
|
2015-05-29 05:40:20 +10:00
|
|
|
CSMWorld::IdTree *tree = static_cast<CSMWorld::IdTree *>(mTable);
|
2015-04-17 01:27:36 +10:00
|
|
|
mNestedTableMapper = new QDataWidgetMapper (this);
|
|
|
|
|
2015-05-29 05:40:20 +10:00
|
|
|
mNestedTableMapper->setModel(tree);
|
2015-04-17 01:27:36 +10:00
|
|
|
// FIXME: lack MIME support?
|
|
|
|
mNestedTableDispatcher =
|
2015-05-29 05:40:20 +10:00
|
|
|
new DialogueDelegateDispatcher (0/*this*/, mTable, mCommandDispatcher, mDocument, tree);
|
|
|
|
mNestedTableMapper->setRootIndex (tree->index(row, i));
|
2015-04-17 01:27:36 +10:00
|
|
|
mNestedTableMapper->setItemDelegate(mNestedTableDispatcher);
|
|
|
|
|
2015-05-29 05:40:20 +10:00
|
|
|
int columnCount = tree->columnCount(tree->index(row, i));
|
2015-04-17 01:27:36 +10:00
|
|
|
for (int col = 0; col < columnCount; ++col)
|
2014-03-09 18:44:04 +01:00
|
|
|
{
|
2015-05-29 05:40:20 +10:00
|
|
|
int displayRole = tree->nestedHeaderData (i, col,
|
2015-04-17 01:27:36 +10:00
|
|
|
Qt::Horizontal, CSMWorld::ColumnBase::Role_Display).toInt();
|
|
|
|
|
|
|
|
CSMWorld::ColumnBase::Display display =
|
|
|
|
static_cast<CSMWorld::ColumnBase::Display> (displayRole);
|
|
|
|
|
|
|
|
mNestedTableDispatcher->makeDelegate (display);
|
|
|
|
|
|
|
|
// FIXME: assumed all columns are editable
|
|
|
|
QWidget* editor =
|
2015-05-29 05:40:20 +10:00
|
|
|
mNestedTableDispatcher->makeEditor (display, tree->index (0, col, tree->index(row, i)));
|
2015-04-17 01:27:36 +10:00
|
|
|
if (editor)
|
|
|
|
{
|
|
|
|
mNestedTableMapper->addMapping (editor, col);
|
|
|
|
|
2015-05-29 05:40:20 +10:00
|
|
|
// Need to use Qt::DisplayRole in order to get the correct string
|
2015-04-17 01:27:36 +10:00
|
|
|
// from CSMWorld::Columns
|
2015-05-29 05:40:20 +10:00
|
|
|
QLabel* label = new QLabel (tree->nestedHeaderData (i, col,
|
2015-04-17 01:27:36 +10:00
|
|
|
Qt::Horizontal, Qt::DisplayRole).toString(), mMainWidget);
|
|
|
|
|
|
|
|
label->setSizePolicy (QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
|
|
editor->setSizePolicy (QSizePolicy::MinimumExpanding, QSizePolicy::Fixed);
|
|
|
|
|
|
|
|
unlockedLayout->addWidget (label, unlocked, 0);
|
|
|
|
unlockedLayout->addWidget (editor, unlocked, 1);
|
|
|
|
++unlocked;
|
2015-05-21 13:11:07 +10:00
|
|
|
|
2015-05-29 05:40:20 +10:00
|
|
|
if(tree->index(0, col, tree->index(row, i)).data().type() == QVariant::UserType)
|
2015-05-21 13:11:07 +10:00
|
|
|
{
|
|
|
|
editor->setEnabled(false);
|
|
|
|
label->setEnabled(false);
|
|
|
|
}
|
2015-07-04 19:55:48 +03:00
|
|
|
|
|
|
|
createEditorContextMenu(editor, display, row);
|
2015-04-17 01:27:36 +10:00
|
|
|
}
|
2014-03-08 15:18:40 +01:00
|
|
|
}
|
2015-05-29 05:40:20 +10:00
|
|
|
mNestedTableMapper->setCurrentModelIndex(tree->index(0, 0, tree->index(row, i)));
|
2013-01-08 14:59:40 +01:00
|
|
|
}
|
2012-12-30 14:01:52 +01:00
|
|
|
}
|
|
|
|
}
|
2013-01-08 14:59:40 +01:00
|
|
|
|
2014-03-12 12:25:37 +01:00
|
|
|
mWidgetMapper->setCurrentModelIndex(mTable->index(row, 0));
|
2014-03-11 21:21:05 +01:00
|
|
|
|
2015-04-19 10:32:06 +10:00
|
|
|
if (unlocked == 0)
|
2015-04-09 19:39:09 +10:00
|
|
|
mainLayout->removeWidget(line);
|
|
|
|
|
2014-03-12 10:21:52 +01:00
|
|
|
this->setWidget(mMainWidget);
|
|
|
|
this->setWidgetResizable(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-26 15:17:47 +02:00
|
|
|
QVBoxLayout& CSVWorld::SimpleDialogueSubView::getMainLayout()
|
|
|
|
{
|
|
|
|
return *mMainLayout;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::IdTable& CSVWorld::SimpleDialogueSubView::getTable()
|
|
|
|
{
|
|
|
|
return *mTable;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSMWorld::CommandDispatcher& CSVWorld::SimpleDialogueSubView::getCommandDispatcher()
|
|
|
|
{
|
|
|
|
return mCommandDispatcher;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSVWorld::EditWidget& CSVWorld::SimpleDialogueSubView::getEditWidget()
|
|
|
|
{
|
|
|
|
return *mEditWidget;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVWorld::SimpleDialogueSubView::isLocked() const
|
|
|
|
{
|
|
|
|
return mLocked;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSVWorld::SimpleDialogueSubView::SimpleDialogueSubView (const CSMWorld::UniversalId& id, CSMDoc::Document& document) :
|
2014-03-12 12:25:37 +01:00
|
|
|
SubView (id),
|
|
|
|
mEditWidget(0),
|
|
|
|
mMainLayout(NULL),
|
|
|
|
mTable(dynamic_cast<CSMWorld::IdTable*>(document.getData().getTableModel(id))),
|
2014-03-13 16:50:04 +01:00
|
|
|
mLocked(false),
|
2014-06-08 14:10:08 +02:00
|
|
|
mDocument(document),
|
|
|
|
mCommandDispatcher (document, CSMWorld::UniversalId::getParentType (id.getType()))
|
2014-03-12 10:21:52 +01:00
|
|
|
{
|
2014-03-13 18:26:59 +01:00
|
|
|
connect(mTable, SIGNAL(dataChanged (const QModelIndex&, const QModelIndex&)), this, SLOT(dataChanged(const QModelIndex&)));
|
2015-05-24 23:19:06 +10:00
|
|
|
connect(mTable, SIGNAL(rowsAboutToBeRemoved(const QModelIndex&, int, int)), this, SLOT(rowsAboutToBeRemoved(const QModelIndex&, int, int)));
|
2014-06-16 11:31:57 +02:00
|
|
|
|
2015-06-27 15:29:54 +02:00
|
|
|
updateCurrentId();
|
2014-06-16 11:31:57 +02:00
|
|
|
|
2014-03-12 11:08:04 +01:00
|
|
|
QWidget *mainWidget = new QWidget(this);
|
|
|
|
|
2015-06-26 15:17:47 +02:00
|
|
|
mMainLayout = new QVBoxLayout(mainWidget);
|
|
|
|
setWidget (mainWidget);
|
|
|
|
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
|
2015-06-26 15:17:47 +02:00
|
|
|
mEditWidget = new EditWidget(mainWidget,
|
2015-08-04 16:56:05 +03:00
|
|
|
mTable->getModelIndex(getUniversalId().getId(), idColumn).row(), mTable, mCommandDispatcher, document, false);
|
2015-06-26 15:17:47 +02:00
|
|
|
|
|
|
|
mMainLayout->addWidget(mEditWidget);
|
|
|
|
mEditWidget->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
|
|
|
|
2015-08-04 16:56:05 +03:00
|
|
|
dataChanged(mTable->getModelIndex (getUniversalId().getId(), idColumn));
|
2015-07-03 16:24:08 +03:00
|
|
|
|
|
|
|
connect(mEditWidget,
|
|
|
|
SIGNAL(editIdRequest(const CSMWorld::UniversalId &, const std::string &)),
|
|
|
|
this,
|
|
|
|
SIGNAL(focusId(const CSMWorld::UniversalId &, const std::string &)));
|
2015-06-26 15:17:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::SimpleDialogueSubView::setEditLock (bool locked)
|
|
|
|
{
|
2015-06-27 15:29:54 +02:00
|
|
|
if (!mEditWidget) // hack to indicate that getUniversalId().getId() is no longer valid
|
2015-06-26 15:17:47 +02:00
|
|
|
return;
|
|
|
|
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
2015-06-26 15:17:47 +02:00
|
|
|
mLocked = locked;
|
2015-08-04 16:56:05 +03:00
|
|
|
QModelIndex currentIndex(mTable->getModelIndex(getUniversalId().getId(), idColumn));
|
2015-06-26 15:17:47 +02:00
|
|
|
|
|
|
|
if (currentIndex.isValid())
|
|
|
|
{
|
|
|
|
CSMWorld::RecordBase::State state = static_cast<CSMWorld::RecordBase::State>(mTable->data (mTable->index (currentIndex.row(), 1)).toInt());
|
|
|
|
|
|
|
|
mEditWidget->setDisabled (state==CSMWorld::RecordBase::State_Deleted || locked);
|
|
|
|
|
|
|
|
mCommandDispatcher.setEditLock (locked);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::SimpleDialogueSubView::dataChanged (const QModelIndex & index)
|
|
|
|
{
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QModelIndex currentIndex(mTable->getModelIndex(getUniversalId().getId(), idColumn));
|
2015-06-26 15:17:47 +02:00
|
|
|
|
|
|
|
if (currentIndex.isValid() &&
|
|
|
|
(index.parent().isValid() ? index.parent().row() : index.row()) == currentIndex.row())
|
|
|
|
{
|
|
|
|
CSMWorld::RecordBase::State state = static_cast<CSMWorld::RecordBase::State>(mTable->data (mTable->index (currentIndex.row(), 1)).toInt());
|
|
|
|
|
|
|
|
mEditWidget->setDisabled (state==CSMWorld::RecordBase::State_Deleted || mLocked);
|
|
|
|
|
|
|
|
// Check if the changed data should force refresh (rebuild) the dialogue subview
|
|
|
|
int flags = 0;
|
|
|
|
if (index.parent().isValid()) // TODO: check that index is topLeft
|
|
|
|
{
|
|
|
|
flags = static_cast<CSMWorld::IdTree *>(mTable)->nestedHeaderData (index.parent().column(),
|
|
|
|
index.column(), Qt::Horizontal, CSMWorld::ColumnBase::Role_Flags).toInt();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
flags = mTable->headerData (index.column(),
|
|
|
|
Qt::Horizontal, CSMWorld::ColumnBase::Role_Flags).toInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & CSMWorld::ColumnBase::Flag_Dialogue_Refresh)
|
|
|
|
{
|
|
|
|
int y = mEditWidget->verticalScrollBar()->value();
|
|
|
|
mEditWidget->remake (index.parent().isValid() ? index.parent().row() : index.row());
|
|
|
|
mEditWidget->verticalScrollBar()->setValue(y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::SimpleDialogueSubView::rowsAboutToBeRemoved(const QModelIndex &parent, int start, int end)
|
|
|
|
{
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QModelIndex currentIndex(mTable->getModelIndex(getUniversalId().getId(), idColumn));
|
2015-08-06 08:45:38 +02:00
|
|
|
|
2015-08-02 22:39:41 +03:00
|
|
|
if (!currentIndex.isValid())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2015-06-26 15:17:47 +02:00
|
|
|
|
2015-08-02 22:39:41 +03:00
|
|
|
if (currentIndex.parent() == parent && currentIndex.row() >= start && currentIndex.row() <= end)
|
2015-06-26 15:17:47 +02:00
|
|
|
{
|
|
|
|
if(mEditWidget)
|
|
|
|
{
|
|
|
|
delete mEditWidget;
|
|
|
|
mEditWidget = 0;
|
|
|
|
}
|
|
|
|
emit closeRequest(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-27 15:29:54 +02:00
|
|
|
void CSVWorld::SimpleDialogueSubView::updateCurrentId()
|
2015-06-26 15:17:47 +02:00
|
|
|
{
|
|
|
|
std::vector<std::string> selection;
|
2015-06-27 15:29:54 +02:00
|
|
|
selection.push_back (getUniversalId().getId());
|
2015-06-26 15:17:47 +02:00
|
|
|
mCommandDispatcher.setSelection(selection);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-13 12:52:18 +02:00
|
|
|
void CSVWorld::DialogueSubView::addButtonBar()
|
|
|
|
{
|
|
|
|
if (mButtons)
|
|
|
|
return;
|
|
|
|
|
|
|
|
mButtons = new RecordButtonBar (getUniversalId(), getTable(), mBottom,
|
|
|
|
&getCommandDispatcher(), this);
|
|
|
|
|
|
|
|
getMainLayout().insertWidget (1, mButtons);
|
|
|
|
|
|
|
|
// connections
|
|
|
|
connect (mButtons, SIGNAL (showPreview()), this, SLOT (showPreview()));
|
|
|
|
connect (mButtons, SIGNAL (viewRecord()), this, SLOT (viewRecord()));
|
|
|
|
connect (mButtons, SIGNAL (switchToRow (int)), this, SLOT (switchToRow (int)));
|
|
|
|
|
|
|
|
connect (this, SIGNAL (universalIdChanged (const CSMWorld::UniversalId&)),
|
|
|
|
mButtons, SLOT (universalIdChanged (const CSMWorld::UniversalId&)));
|
|
|
|
}
|
|
|
|
|
2015-06-26 15:17:47 +02:00
|
|
|
CSVWorld::DialogueSubView::DialogueSubView (const CSMWorld::UniversalId& id,
|
|
|
|
CSMDoc::Document& document, const CreatorFactoryBase& creatorFactory, bool sorting)
|
2015-07-13 12:52:18 +02:00
|
|
|
: SimpleDialogueSubView (id, document), mButtons (0)
|
2015-06-26 15:17:47 +02:00
|
|
|
{
|
|
|
|
// bottom box
|
2015-06-27 12:53:46 +02:00
|
|
|
mBottom = new TableBottomBox (creatorFactory, document, id, this);
|
2015-06-26 15:17:47 +02:00
|
|
|
|
2015-06-27 12:53:46 +02:00
|
|
|
connect (mBottom, SIGNAL (requestFocus (const std::string&)),
|
|
|
|
this, SLOT (requestFocus (const std::string&)));
|
2015-06-26 15:17:47 +02:00
|
|
|
|
2015-06-27 12:53:46 +02:00
|
|
|
// layout
|
|
|
|
getMainLayout().addWidget (mBottom);
|
2015-12-15 10:40:00 +01:00
|
|
|
|
|
|
|
connect (&CSMPrefs::State::get(), SIGNAL (settingChanged (const CSMPrefs::Setting *)),
|
|
|
|
this, SLOT (settingChanged (const CSMPrefs::Setting *)));
|
|
|
|
CSMPrefs::get()["ID Dialogues"].update();
|
2015-06-27 14:25:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::DialogueSubView::setEditLock (bool locked)
|
|
|
|
{
|
|
|
|
SimpleDialogueSubView::setEditLock (locked);
|
2015-07-13 12:52:18 +02:00
|
|
|
|
|
|
|
if (mButtons)
|
|
|
|
mButtons->setEditLock (locked);
|
2015-06-26 15:17:47 +02:00
|
|
|
}
|
|
|
|
|
2015-12-15 10:40:00 +01:00
|
|
|
void CSVWorld::DialogueSubView::settingChanged (const CSMPrefs::Setting *setting)
|
2015-06-27 16:57:45 +02:00
|
|
|
{
|
2015-12-15 10:40:00 +01:00
|
|
|
if (*setting=="ID Dialogues/toolbar")
|
2015-07-13 12:52:18 +02:00
|
|
|
{
|
2015-12-15 10:40:00 +01:00
|
|
|
if (setting->isTrue())
|
2015-07-13 12:52:18 +02:00
|
|
|
{
|
|
|
|
addButtonBar();
|
|
|
|
}
|
2015-12-15 10:40:00 +01:00
|
|
|
else if (mButtons)
|
2015-07-13 12:52:18 +02:00
|
|
|
{
|
2015-12-15 10:40:00 +01:00
|
|
|
getMainLayout().removeWidget (mButtons);
|
|
|
|
delete mButtons;
|
|
|
|
mButtons = 0;
|
2015-07-13 12:52:18 +02:00
|
|
|
}
|
|
|
|
}
|
2015-06-27 16:57:45 +02:00
|
|
|
}
|
|
|
|
|
2014-03-19 11:42:43 +01:00
|
|
|
void CSVWorld::DialogueSubView::showPreview ()
|
|
|
|
{
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QModelIndex currentIndex (getTable().getModelIndex (getUniversalId().getId(), idColumn));
|
2014-06-09 11:37:48 +02:00
|
|
|
|
|
|
|
if (currentIndex.isValid() &&
|
2015-06-26 15:17:47 +02:00
|
|
|
getTable().getFeatures() & CSMWorld::IdTable::Feature_Preview &&
|
|
|
|
currentIndex.row() < getTable().rowCount())
|
2014-03-19 11:42:43 +01:00
|
|
|
{
|
2015-06-27 15:29:54 +02:00
|
|
|
emit focusId(CSMWorld::UniversalId(CSMWorld::UniversalId::Type_Preview, getUniversalId().getId()), "");
|
2014-03-19 11:42:43 +01:00
|
|
|
}
|
2014-03-19 12:01:36 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 10:46:54 +02:00
|
|
|
void CSVWorld::DialogueSubView::viewRecord ()
|
2014-03-19 12:01:36 +01:00
|
|
|
{
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QModelIndex currentIndex (getTable().getModelIndex (getUniversalId().getId(), idColumn));
|
2014-06-09 11:37:48 +02:00
|
|
|
|
|
|
|
if (currentIndex.isValid() &&
|
2015-06-26 15:17:47 +02:00
|
|
|
currentIndex.row() < getTable().rowCount())
|
2014-03-19 12:01:36 +01:00
|
|
|
{
|
2015-06-26 15:17:47 +02:00
|
|
|
std::pair<CSMWorld::UniversalId, std::string> params = getTable().view (currentIndex.row());
|
2014-03-19 12:01:36 +01:00
|
|
|
|
|
|
|
if (params.first.getType()!=CSMWorld::UniversalId::Type_None)
|
|
|
|
emit focusId (params.first, params.second);
|
|
|
|
}
|
2014-03-21 13:44:01 +01:00
|
|
|
}
|
2015-06-27 14:42:22 +02:00
|
|
|
|
|
|
|
void CSVWorld::DialogueSubView::switchToRow (int row)
|
|
|
|
{
|
2015-06-27 15:02:50 +02:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
std::string id = getTable().data (getTable().index (row, idColumn)).toString().toUtf8().constData();
|
|
|
|
|
|
|
|
int typeColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_RecordType);
|
|
|
|
CSMWorld::UniversalId::Type type = static_cast<CSMWorld::UniversalId::Type> (
|
|
|
|
getTable().data (getTable().index (row, typeColumn)).toInt());
|
2015-06-27 14:42:22 +02:00
|
|
|
|
2015-06-27 15:02:50 +02:00
|
|
|
setUniversalId (CSMWorld::UniversalId (type, id));
|
2015-06-27 15:29:54 +02:00
|
|
|
updateCurrentId();
|
2015-07-13 12:52:18 +02:00
|
|
|
|
2015-06-27 15:02:50 +02:00
|
|
|
getEditWidget().remake (row);
|
2015-06-27 14:42:22 +02:00
|
|
|
|
2015-06-27 15:02:50 +02:00
|
|
|
int stateColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Modification);
|
|
|
|
CSMWorld::RecordBase::State state = static_cast<CSMWorld::RecordBase::State> (
|
|
|
|
getTable().data (getTable().index (row, stateColumn)).toInt());
|
2015-06-27 14:42:22 +02:00
|
|
|
|
2015-06-27 15:02:50 +02:00
|
|
|
getEditWidget().setDisabled (isLocked() || state==CSMWorld::RecordBase::State_Deleted);
|
2015-06-27 14:42:22 +02:00
|
|
|
}
|
2015-06-27 15:29:54 +02:00
|
|
|
|
|
|
|
void CSVWorld::DialogueSubView::requestFocus (const std::string& id)
|
|
|
|
{
|
2015-08-04 16:56:05 +03:00
|
|
|
int idColumn = getTable().findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
QModelIndex index = getTable().getModelIndex (id, idColumn);
|
2015-06-27 15:29:54 +02:00
|
|
|
|
|
|
|
if (index.isValid())
|
2015-07-13 12:52:18 +02:00
|
|
|
switchToRow (index.row());
|
2015-06-27 15:29:54 +02:00
|
|
|
}
|