1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-21 08:53:52 +00:00
openmw/apps/opencs/view/world/dialoguesubview.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

983 lines
34 KiB
C++
Raw Normal View History

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