#include "tablesubview.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../../model/doc/document.hpp" #include "../../model/world/tablemimedata.hpp" #include "../doc/sizehint.hpp" #include "../filter/filterbox.hpp" #include "../filter/filterdata.hpp" #include "table.hpp" #include "tablebottombox.hpp" CSVWorld::TableSubView::TableSubView( const CSMWorld::UniversalId& id, CSMDoc::Document& document, const CreatorFactoryBase& creatorFactory, bool sorting) : SubView(id) , mShowOptions(false) , mOptions(0) { QVBoxLayout* layout = new QVBoxLayout; layout->addWidget(mBottom = new TableBottomBox(creatorFactory, document, id, this), 0); layout->insertWidget(0, mTable = new Table(id, mBottom->canCreateAndDelete(), sorting, document), 2); mFilterBox = new CSVFilter::FilterBox(document.getData(), this); QHBoxLayout* hLayout = new QHBoxLayout; hLayout->insertWidget(0, mFilterBox); mOptions = new QWidget; QHBoxLayout* optHLayout = new QHBoxLayout; QCheckBox* autoJump = new QCheckBox("Auto Jump"); autoJump->setToolTip( "Whether to jump to the modified record." "\nCan be useful in finding the moved or modified" "\nobject instance while 3D editing."); autoJump->setCheckState(Qt::Unchecked); connect(autoJump, &QCheckBox::stateChanged, mTable, &Table::jumpAfterModChanged); optHLayout->insertWidget(0, autoJump); optHLayout->setContentsMargins(QMargins(0, 3, 0, 0)); mOptions->setLayout(optHLayout); mOptions->resize(mOptions->width(), mFilterBox->height()); mOptions->hide(); QPushButton* opt = new QPushButton(); opt->setIcon(QIcon(":startup/configure")); opt->setSizePolicy(QSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed)); opt->setToolTip("Open additional options for this subview."); connect(opt, &QPushButton::clicked, this, &TableSubView::toggleOptions); QVBoxLayout* buttonLayout = new QVBoxLayout; // work around margin issues buttonLayout->setContentsMargins(QMargins(0 /*left*/, 3 /*top*/, 3 /*right*/, 0 /*bottom*/)); buttonLayout->insertWidget(0, opt, 0, Qt::AlignVCenter | Qt::AlignRight); hLayout->insertWidget(1, mOptions); hLayout->insertLayout(2, buttonLayout); layout->insertLayout(0, hLayout); CSVDoc::SizeHintWidget* widget = new CSVDoc::SizeHintWidget; widget->setLayout(layout); setWidget(widget); // Widget position can be negative, we should clamp it. QPoint position = pos(); if (position.x() <= 0) position.setX(0); if (position.y() <= 0) position.setY(0); // prefer height of the screen and full width of the table const QRect rect = QApplication::screenAt(position)->geometry(); int frameHeight = 40; // set a reasonable default QWidget* topLevel = QApplication::topLevelAt(pos()); if (topLevel) frameHeight = topLevel->frameGeometry().height() - topLevel->height(); widget->setSizeHint(QSize(mTable->horizontalHeader()->length(), rect.height() - frameHeight)); connect(mTable, &Table::editRequest, this, &TableSubView::editRequest); connect(mTable, &Table::selectionSizeChanged, mBottom, &TableBottomBox::selectionSizeChanged); connect(mTable, &Table::tableSizeChanged, mBottom, &TableBottomBox::tableSizeChanged); mTable->tableSizeUpdate(); mTable->selectionSizeUpdate(); mTable->viewport()->installEventFilter(this); mBottom->installEventFilter(this); mFilterBox->installEventFilter(this); if (mBottom->canCreateAndDelete()) { connect(mTable, &Table::createRequest, mBottom, &TableBottomBox::createRequest); connect( mTable, &Table::cloneRequest, this, qOverload(&TableSubView::cloneRequest)); connect(this, qOverload(&TableSubView::cloneRequest), mBottom, &TableBottomBox::cloneRequest); connect(mTable, &Table::createRecordsDirectlyRequest, mBottom, &TableBottomBox::createRecordsDirectlyRequest); connect(mTable, &Table::touchRequest, mBottom, &TableBottomBox::touchRequest); connect(mTable, &Table::extendedDeleteConfigRequest, mBottom, &TableBottomBox::extendedDeleteConfigRequest); connect(mTable, &Table::extendedRevertConfigRequest, mBottom, &TableBottomBox::extendedRevertConfigRequest); } connect(mBottom, &TableBottomBox::requestFocus, mTable, &Table::requestFocus); connect(mFilterBox, &CSVFilter::FilterBox::recordFilterChanged, mTable, &Table::recordFilterChanged); connect(mFilterBox, &CSVFilter::FilterBox::recordDropped, this, &TableSubView::createFilterRequest); connect(mTable, &Table::closeRequest, this, qOverload<>(&TableSubView::closeRequest)); } void CSVWorld::TableSubView::setEditLock(bool locked) { mTable->setEditLock(locked); mBottom->setEditLock(locked); } void CSVWorld::TableSubView::editRequest(const CSMWorld::UniversalId& id, const std::string& hint) { focusId(id, hint); } void CSVWorld::TableSubView::setStatusBar(bool show) { mBottom->setStatusBar(show); } void CSVWorld::TableSubView::useHint(const std::string& hint) { if (hint.empty()) return; if (hint[0] == 'f' && hint.size() >= 2) mFilterBox->setRecordFilter(hint.substr(2)); } void CSVWorld::TableSubView::cloneRequest(const CSMWorld::UniversalId& toClone) { emit cloneRequest(toClone.getId(), toClone.getType()); } void CSVWorld::TableSubView::createFilterRequest(std::vector& types, const std::pair& columnSearchData, Qt::DropAction action) { std::vector sourceFilter; std::vector refIdColumns = mTable->getColumnsWithDisplay( CSMWorld::TableMimeData::convertEnums(CSMWorld::UniversalId::Type_Referenceable)); bool hasRefIdDisplay = !refIdColumns.empty(); for (std::vector::iterator it(types.begin()); it != types.end(); ++it) { CSMWorld::UniversalId::Type type = it->getType(); std::vector col = mTable->getColumnsWithDisplay(CSMWorld::TableMimeData::convertEnums(type)); if (!col.empty()) { CSVFilter::FilterData filterData; filterData.searchData = it->getId(); filterData.columns = std::move(col); sourceFilter.emplace_back(filterData); } if (hasRefIdDisplay && CSMWorld::TableMimeData::isReferencable(type)) { CSVFilter::FilterData filterData; filterData.searchData = it->getId(); filterData.columns = refIdColumns; sourceFilter.emplace_back(filterData); } } if (!sourceFilter.empty()) mFilterBox->createFilterRequest(sourceFilter, action); else { std::vector sourceFilterByValue; QVariant qData = columnSearchData.first; std::string searchColumn = columnSearchData.second; std::vector searchColumns; searchColumns.emplace_back(searchColumn); CSVFilter::FilterData filterData; filterData.searchData = qData; filterData.columns = std::move(searchColumns); sourceFilterByValue.emplace_back(filterData); mFilterBox->createFilterRequest(sourceFilterByValue, action); } } bool CSVWorld::TableSubView::eventFilter(QObject* object, QEvent* event) { if (event->type() == QEvent::Drop) { if (QDropEvent* drop = dynamic_cast(event)) { const CSMWorld::TableMimeData* tableMimeData = dynamic_cast(drop->mimeData()); if (!tableMimeData) // May happen when non-records (e.g. plain text) are dragged and dropped return false; bool handled = tableMimeData->holdsType(CSMWorld::UniversalId::Type_Filter); if (handled) { mFilterBox->setRecordFilter(tableMimeData->returnMatching(CSMWorld::UniversalId::Type_Filter).getId()); } return handled; } } return false; } void CSVWorld::TableSubView::toggleOptions() { if (mShowOptions) { mShowOptions = false; mOptions->hide(); } else { mShowOptions = true; mOptions->show(); } } void CSVWorld::TableSubView::requestFocus(const std::string& id) { mTable->requestFocus(id); }