mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 08:56:37 +00:00 
			
		
		
		
	I tried to fix https://gitlab.com/OpenMW/openmw/-/issues/8080 by making it so that instead of crashing, we showed an error. In doing so, I discovered some problems with plugin sorting and the refresh button, like: * it forgetting the non-user content files somewhere * nothing guaranteeing that built-in content files stay at the top of the list and them only being there because the first data directory that provides them is usually the first data directory * it forgetting the non-user content files somewhere else * it looking like it'd forget any kind of non-user setting under certain circumstances I fixed those problems too
		
			
				
	
	
		
			345 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "contentselector.hpp"
 | |
| 
 | |
| #include "ui_contentselector.h"
 | |
| 
 | |
| #include <components/contentselector/model/esmfile.hpp>
 | |
| 
 | |
| #include <QClipboard>
 | |
| #include <QMenu>
 | |
| #include <QModelIndex>
 | |
| #include <QSortFilterProxyModel>
 | |
| 
 | |
| ContentSelectorView::ContentSelector::ContentSelector(QWidget* parent, bool showOMWScripts)
 | |
|     : QObject(parent)
 | |
|     , ui(std::make_unique<Ui::ContentSelector>())
 | |
| {
 | |
|     ui->setupUi(parent);
 | |
|     ui->addonView->setDragDropMode(QAbstractItemView::InternalMove);
 | |
| 
 | |
|     if (!showOMWScripts)
 | |
|     {
 | |
|         ui->languageComboBox->setHidden(true);
 | |
|         ui->refreshButton->setHidden(true);
 | |
|     }
 | |
| 
 | |
|     buildContentModel(showOMWScripts);
 | |
|     buildGameFileView();
 | |
|     buildAddonView();
 | |
| }
 | |
| 
 | |
| ContentSelectorView::ContentSelector::~ContentSelector() = default;
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::buildContentModel(bool showOMWScripts)
 | |
| {
 | |
|     QIcon warningIcon(ui->addonView->style()->standardIcon(QStyle::SP_MessageBoxWarning));
 | |
|     QIcon errorIcon(ui->addonView->style()->standardIcon(QStyle::SP_MessageBoxCritical));
 | |
|     mContentModel = new ContentSelectorModel::ContentModel(this, warningIcon, errorIcon, showOMWScripts);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::buildGameFileView()
 | |
| {
 | |
|     ui->gameFileView->addItem(tr("<No game file>"));
 | |
|     ui->gameFileView->setVisible(true);
 | |
| 
 | |
|     connect(ui->gameFileView, qOverload<int>(&ComboBox::currentIndexChanged), this,
 | |
|         &ContentSelector::slotCurrentGameFileIndexChanged);
 | |
| 
 | |
|     ui->gameFileView->setCurrentIndex(0);
 | |
| }
 | |
| 
 | |
| class AddOnProxyModel : public QSortFilterProxyModel
 | |
| {
 | |
| public:
 | |
|     explicit AddOnProxyModel(QObject* parent = nullptr)
 | |
|         : QSortFilterProxyModel(parent)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     bool filterAcceptsRow(int sourceRow, const QModelIndex& sourceParent) const override
 | |
|     {
 | |
|         static const QString ContentTypeAddon = QString::number((int)ContentSelectorModel::ContentType_Addon);
 | |
| 
 | |
|         QModelIndex nameIndex = sourceModel()->index(sourceRow, 0, sourceParent);
 | |
|         const QString userRole = sourceModel()->data(nameIndex, Qt::UserRole).toString();
 | |
| 
 | |
|         return QSortFilterProxyModel::filterAcceptsRow(sourceRow, sourceParent) && userRole == ContentTypeAddon;
 | |
|     }
 | |
| };
 | |
| 
 | |
| bool ContentSelectorView::ContentSelector::isGamefileSelected() const
 | |
| {
 | |
|     return ui->gameFileView->currentIndex() > 0;
 | |
| }
 | |
| 
 | |
| QWidget* ContentSelectorView::ContentSelector::uiWidget() const
 | |
| {
 | |
|     return ui->contentGroupBox;
 | |
| }
 | |
| 
 | |
| QComboBox* ContentSelectorView::ContentSelector::languageBox() const
 | |
| {
 | |
|     return ui->languageComboBox;
 | |
| }
 | |
| 
 | |
| QToolButton* ContentSelectorView::ContentSelector::refreshButton() const
 | |
| {
 | |
|     return ui->refreshButton;
 | |
| }
 | |
| 
 | |
| QLineEdit* ContentSelectorView::ContentSelector::searchFilter() const
 | |
| {
 | |
|     return ui->searchFilter;
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::buildAddonView()
 | |
| {
 | |
|     ui->addonView->setVisible(true);
 | |
| 
 | |
|     mAddonProxyModel = new AddOnProxyModel(this);
 | |
|     mAddonProxyModel->setFilterRegularExpression(searchFilter()->text());
 | |
|     mAddonProxyModel->setFilterCaseSensitivity(Qt::CaseInsensitive);
 | |
|     mAddonProxyModel->setDynamicSortFilter(true);
 | |
|     mAddonProxyModel->setSourceModel(mContentModel);
 | |
| 
 | |
|     connect(ui->searchFilter, &QLineEdit::textEdited, mAddonProxyModel, &QSortFilterProxyModel::setFilterWildcard);
 | |
|     connect(ui->searchFilter, &QLineEdit::textEdited, this, &ContentSelector::slotSearchFilterTextChanged);
 | |
| 
 | |
|     ui->addonView->setModel(mAddonProxyModel);
 | |
| 
 | |
|     connect(ui->addonView, &QTableView::activated, this, &ContentSelector::slotAddonTableItemActivated);
 | |
|     connect(mContentModel, &ContentSelectorModel::ContentModel::dataChanged, this,
 | |
|         &ContentSelector::signalAddonDataChanged);
 | |
|     connect(mContentModel, &ContentSelectorModel::ContentModel::dataChanged, this, &ContentSelector::slotRowsMoved);
 | |
|     buildContextMenu();
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::buildContextMenu()
 | |
| {
 | |
|     ui->addonView->setContextMenuPolicy(Qt::CustomContextMenu);
 | |
|     connect(ui->addonView, &QTableView::customContextMenuRequested, this, &ContentSelector::slotShowContextMenu);
 | |
| 
 | |
|     mContextMenu = new QMenu(ui->addonView);
 | |
|     mContextMenu->addAction(tr("&Check Selected"), this, SLOT(slotCheckMultiSelectedItems()));
 | |
|     mContextMenu->addAction(tr("&Uncheck Selected"), this, SLOT(slotUncheckMultiSelectedItems()));
 | |
|     mContextMenu->addAction(tr("&Copy Path(s) to Clipboard"), this, SLOT(slotCopySelectedItemsPaths()));
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setNonUserContent(const QStringList& fileList)
 | |
| {
 | |
|     mContentModel->setNonUserContent(fileList);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setProfileContent(const QStringList& fileList)
 | |
| {
 | |
|     clearCheckStates();
 | |
| 
 | |
|     for (const QString& filepath : fileList)
 | |
|     {
 | |
|         const ContentSelectorModel::EsmFile* file = mContentModel->item(filepath);
 | |
|         if (file && file->isGameFile())
 | |
|         {
 | |
|             setGameFile(filepath);
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     setContentList(fileList);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setGameFile(const QString& filename)
 | |
| {
 | |
|     int index = 0;
 | |
| 
 | |
|     if (!filename.isEmpty())
 | |
|     {
 | |
|         const ContentSelectorModel::EsmFile* file = mContentModel->item(filename);
 | |
|         index = ui->gameFileView->findText(file->fileName());
 | |
| 
 | |
|         // verify that the current index is also checked in the model
 | |
|         if (!mContentModel->setCheckState(filename, true))
 | |
|         {
 | |
|             // throw error in case file not found?
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     ui->gameFileView->setCurrentIndex(index);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::clearCheckStates()
 | |
| {
 | |
|     mContentModel->uncheckAll();
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setEncoding(const QString& encoding)
 | |
| {
 | |
|     mContentModel->setEncoding(encoding);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setContentList(const QStringList& list)
 | |
| {
 | |
|     if (list.isEmpty())
 | |
|     {
 | |
|         slotCurrentGameFileIndexChanged(ui->gameFileView->currentIndex());
 | |
|     }
 | |
|     else
 | |
|         mContentModel->setContentList(list);
 | |
| }
 | |
| 
 | |
| ContentSelectorModel::ContentFileList ContentSelectorView::ContentSelector::selectedFiles() const
 | |
| {
 | |
|     if (!mContentModel)
 | |
|         return ContentSelectorModel::ContentFileList();
 | |
| 
 | |
|     return mContentModel->checkedItems();
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::addFiles(const QString& path, bool newfiles)
 | |
| {
 | |
|     mContentModel->addFiles(path, newfiles);
 | |
| 
 | |
|     // add any game files to the combo box
 | |
|     for (const QString& gameFileName : mContentModel->gameFiles())
 | |
|     {
 | |
|         if (ui->gameFileView->findText(gameFileName) == -1)
 | |
|         {
 | |
|             ui->gameFileView->addItem(gameFileName);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (ui->gameFileView->currentIndex() != 0)
 | |
|         ui->gameFileView->setCurrentIndex(0);
 | |
| 
 | |
|     mContentModel->uncheckAll();
 | |
|     mContentModel->checkForLoadOrderErrors();
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::sortFiles()
 | |
| {
 | |
|     mContentModel->sortFiles();
 | |
| }
 | |
| 
 | |
| bool ContentSelectorView::ContentSelector::containsDataFiles(const QString& path)
 | |
| {
 | |
|     return mContentModel->containsDataFiles(path);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::clearFiles()
 | |
| {
 | |
|     mContentModel->clearFiles();
 | |
| }
 | |
| 
 | |
| QString ContentSelectorView::ContentSelector::currentFile() const
 | |
| {
 | |
|     QModelIndex currentIdx = ui->addonView->currentIndex();
 | |
| 
 | |
|     if (!currentIdx.isValid() && ui->gameFileView->currentIndex() > 0)
 | |
|         return ui->gameFileView->currentText();
 | |
| 
 | |
|     QModelIndex idx = mContentModel->index(mAddonProxyModel->mapToSource(currentIdx).row(), 0, QModelIndex());
 | |
|     return mContentModel->data(idx, Qt::DisplayRole).toString();
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotCurrentGameFileIndexChanged(int index)
 | |
| {
 | |
|     static int oldIndex = -1;
 | |
| 
 | |
|     if (index != oldIndex)
 | |
|     {
 | |
|         if (oldIndex > -1)
 | |
|         {
 | |
|             setGameFileSelected(oldIndex, false);
 | |
|         }
 | |
| 
 | |
|         oldIndex = index;
 | |
| 
 | |
|         setGameFileSelected(index, true);
 | |
|         mContentModel->checkForLoadOrderErrors();
 | |
|     }
 | |
| 
 | |
|     emit signalCurrentGamefileIndexChanged(index);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setGameFileSelected(int index, bool selected)
 | |
| {
 | |
|     QString fileName = ui->gameFileView->itemText(index);
 | |
|     const ContentSelectorModel::EsmFile* file = mContentModel->item(fileName);
 | |
|     if (file != nullptr)
 | |
|     {
 | |
|         QModelIndex index2(mContentModel->indexFromItem(file));
 | |
|         mContentModel->setData(index2, selected, Qt::UserRole + 1);
 | |
|     }
 | |
|     mContentModel->setCurrentGameFile(selected ? file : nullptr);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotAddonTableItemActivated(const QModelIndex& index)
 | |
| {
 | |
|     // toggles check state when an AddOn file is double clicked or activated by keyboard
 | |
|     QModelIndex sourceIndex = mAddonProxyModel->mapToSource(index);
 | |
| 
 | |
|     if (!mContentModel->isEnabled(sourceIndex))
 | |
|         return;
 | |
| 
 | |
|     Qt::CheckState checkState = Qt::Unchecked;
 | |
| 
 | |
|     if (mContentModel->data(sourceIndex, Qt::CheckStateRole).toInt() == Qt::Unchecked)
 | |
|         checkState = Qt::Checked;
 | |
| 
 | |
|     mContentModel->setData(sourceIndex, checkState, Qt::CheckStateRole);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotShowContextMenu(const QPoint& pos)
 | |
| {
 | |
|     QPoint globalPos = ui->addonView->viewport()->mapToGlobal(pos);
 | |
|     mContextMenu->exec(globalPos);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::setCheckStateForMultiSelectedItems(bool checked)
 | |
| {
 | |
|     Qt::CheckState checkState = checked ? Qt::Checked : Qt::Unchecked;
 | |
|     for (const QModelIndex& index : ui->addonView->selectionModel()->selectedIndexes())
 | |
|     {
 | |
|         QModelIndex sourceIndex = mAddonProxyModel->mapToSource(index);
 | |
|         if (mContentModel->data(sourceIndex, Qt::CheckStateRole).toInt() != checkState)
 | |
|         {
 | |
|             mContentModel->setData(sourceIndex, checkState, Qt::CheckStateRole);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotUncheckMultiSelectedItems()
 | |
| {
 | |
|     setCheckStateForMultiSelectedItems(false);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotCheckMultiSelectedItems()
 | |
| {
 | |
|     setCheckStateForMultiSelectedItems(true);
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotCopySelectedItemsPaths()
 | |
| {
 | |
|     QClipboard* clipboard = QApplication::clipboard();
 | |
|     QString filepaths;
 | |
|     for (const QModelIndex& index : ui->addonView->selectionModel()->selectedIndexes())
 | |
|     {
 | |
|         int row = mAddonProxyModel->mapToSource(index).row();
 | |
|         const ContentSelectorModel::EsmFile* file = mContentModel->item(row);
 | |
|         filepaths += file->filePath() + "\n";
 | |
|     }
 | |
| 
 | |
|     if (!filepaths.isEmpty())
 | |
|     {
 | |
|         clipboard->setText(filepaths);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotSearchFilterTextChanged(const QString& newText)
 | |
| {
 | |
|     ui->addonView->setDragEnabled(newText.isEmpty());
 | |
| }
 | |
| 
 | |
| void ContentSelectorView::ContentSelector::slotRowsMoved()
 | |
| {
 | |
|     ui->addonView->selectionModel()->clearSelection();
 | |
| }
 |