#include #include // TODO: Remove #include #include #include #include #include "datafilespage.hpp" #include "lineedit.hpp" using namespace ESM; using namespace std; DataFilesPage::DataFilesPage(QWidget *parent) : QWidget(parent) { mDataFilesModel = new QStandardItemModel(); // Contains all plugins with masters mPluginsModel = new QStandardItemModel(); // Contains selectable plugins mPluginsProxyModel = new QSortFilterProxyModel(); mPluginsProxyModel->setDynamicSortFilter(true); mPluginsProxyModel->setSourceModel(mPluginsModel); // TODO: decide what to do with the selection model mPluginsSelectModel = new QItemSelectionModel(mPluginsModel); QLabel *filterLabel = new QLabel(tr("Filter:"), this); LineEdit *filterLineEdit = new LineEdit(this); QHBoxLayout *topLayout = new QHBoxLayout(); QSpacerItem *hSpacer1 = new QSpacerItem(40, 20, QSizePolicy::Expanding, QSizePolicy::Minimum); topLayout->addItem(hSpacer1); topLayout->addWidget(filterLabel); topLayout->addWidget(filterLineEdit); mMastersWidget = new QTableWidget(this); // Contains the available masters mPluginsTable = new QTableView(this); mMastersWidget->setObjectName("MastersWidget"); mMastersWidget->setSelectionBehavior(QAbstractItemView::SelectRows); mMastersWidget->setSelectionMode(QAbstractItemView::MultiSelection); mMastersWidget->setEditTriggers(QAbstractItemView::NoEditTriggers); mMastersWidget->setAlternatingRowColors(true); mMastersWidget->horizontalHeader()->setStretchLastSection(true); mMastersWidget->horizontalHeader()->hide(); mMastersWidget->verticalHeader()->hide(); mMastersWidget->insertColumn(0); mPluginsTable->setModel(mPluginsProxyModel); mPluginsTable->setSelectionModel(mPluginsSelectModel); mPluginsTable->setSelectionBehavior(QAbstractItemView::SelectRows); mPluginsTable->setSelectionMode(QAbstractItemView::ExtendedSelection); mPluginsTable->setEditTriggers(QAbstractItemView::NoEditTriggers); mPluginsTable->setAlternatingRowColors(true); mPluginsTable->horizontalHeader()->setStretchLastSection(true); mPluginsTable->horizontalHeader()->hide(); mPluginsTable->setDragEnabled(true); mPluginsTable->setDragDropMode(QAbstractItemView::InternalMove); mPluginsTable->setDropIndicatorShown(true); mPluginsTable->setDragDropOverwriteMode(false); mPluginsTable->viewport()->setAcceptDrops(true); mPluginsTable->setContextMenuPolicy(Qt::CustomContextMenu); // Add both tables to a splitter QSplitter *splitter = new QSplitter(this); splitter->setOrientation(Qt::Horizontal); splitter->addWidget(mMastersWidget); splitter->addWidget(mPluginsTable); // Adjust the default widget widths inside the splitter QList sizeList; sizeList << 100 << 300; splitter->setSizes(sizeList); // Bottom part with profile options QLabel *profileLabel = new QLabel(tr("Current Profile:"), this); mProfilesComboBox = new ComboBox(this); mProfilesComboBox->setSizePolicy(QSizePolicy(QSizePolicy::Expanding, QSizePolicy::Minimum)); mProfilesComboBox->setInsertPolicy(QComboBox::InsertAtBottom); mNewProfileButton = new QPushButton(this); mNewProfileButton->setIcon(QIcon::fromTheme("document-new")); mNewProfileButton->setToolTip(tr("New Profile")); mNewProfileButton->setShortcut(QKeySequence(tr("Ctrl+N"))); mCopyProfileButton = new QPushButton(this); mCopyProfileButton->setIcon(QIcon::fromTheme("edit-copy")); mCopyProfileButton->setToolTip(tr("Copy Profile")); mDeleteProfileButton = new QPushButton(this); mDeleteProfileButton->setIcon(QIcon::fromTheme("edit-delete")); mDeleteProfileButton->setToolTip(tr("Delete Profile")); mDeleteProfileButton->setShortcut(QKeySequence(tr("Delete"))); QHBoxLayout *bottomLayout = new QHBoxLayout(); bottomLayout->addWidget(profileLabel); bottomLayout->addWidget(mProfilesComboBox); bottomLayout->addWidget(mNewProfileButton); bottomLayout->addWidget(mCopyProfileButton); bottomLayout->addWidget(mDeleteProfileButton); QVBoxLayout *pageLayout = new QVBoxLayout(this); // Add some space above and below the page items QSpacerItem *vSpacer2 = new QSpacerItem(5, 5, QSizePolicy::Minimum, QSizePolicy::Minimum); QSpacerItem *vSpacer3 = new QSpacerItem(5, 5, QSizePolicy::Minimum, QSizePolicy::Minimum); pageLayout->addLayout(topLayout); pageLayout->addItem(vSpacer2); pageLayout->addWidget(splitter); pageLayout->addLayout(bottomLayout); pageLayout->addItem(vSpacer3); connect(mMastersWidget->selectionModel(), SIGNAL(selectionChanged(const QItemSelection&, const QItemSelection&)), this, SLOT(masterSelectionChanged(const QItemSelection&, const QItemSelection&))); connect(filterLineEdit, SIGNAL(textChanged(QString)), this, SLOT(filterChanged(const QString))); connect(mPluginsTable, SIGNAL(doubleClicked(QModelIndex)), this, SLOT(setCheckstate(QModelIndex))); connect(mPluginsModel, SIGNAL(dataChanged(const QModelIndex&, const QModelIndex&)), this, SLOT(resizeRows())); connect(mNewProfileButton, SIGNAL(pressed()), this, SLOT(newProfile())); connect(mCopyProfileButton, SIGNAL(pressed()), this, SLOT(copyProfile())); connect(mDeleteProfileButton, SIGNAL(pressed()), this, SLOT(deleteProfile())); setupConfig(); } void DataFilesPage::setupDataFiles(const QStringList &paths, bool strict) { qDebug() << "setupDataFiles called!"; // Put the paths in a boost::filesystem vector to use with Files::Collections std::vector dataDirs; foreach (const QString ¤tPath, paths) { dataDirs.push_back(boost::filesystem::path(currentPath.toStdString())); } // Create a file collection for the dataDirs Files::Collections mFileCollections(dataDirs, strict); // First we add all the master files const Files::MultiDirCollection &esm = mFileCollections.getCollection(".esm"); unsigned int i = 0; // Row number for (Files::MultiDirCollection::TIter iter(esm.begin()); iter!=esm.end(); ++iter) { qDebug() << "Master: " << QString::fromStdString(iter->second.filename().string()); QString currentMaster = QString::fromStdString(iter->second.filename().string()); const QList itemList = mMastersWidget->findItems(currentMaster, Qt::MatchExactly); if (itemList.isEmpty()) // Master is not yet in the widget { qDebug() << "Master not yet in the widget, rowcount is " << i; mMastersWidget->insertRow(i); QTableWidgetItem *item = new QTableWidgetItem(currentMaster); mMastersWidget->setItem(i, 0, item); ++i; } } // Now on to the plugins const Files::MultiDirCollection &esp = mFileCollections.getCollection(".esp"); for (Files::MultiDirCollection::TIter iter(esp.begin()); iter!=esp.end(); ++iter) { ESMReader fileReader; QStringList availableMasters; // Will contain all found masters fileReader.open(iter->second.string()); // First we fill the availableMasters and the mMastersWidget ESMReader::MasterList mlist = fileReader.getMasters(); for (unsigned int i = 0; i < mlist.size(); ++i) { const QString currentMaster = QString::fromStdString(mlist[i].name); availableMasters.append(currentMaster); const QList itemList = mMastersWidget->findItems(currentMaster, Qt::MatchExactly); if (itemList.isEmpty()) // Master is not yet in the widget { // TODO: Show warning, missing master mMastersWidget->insertRow(i); QTableWidgetItem *item = new QTableWidgetItem(currentMaster); mMastersWidget->setItem(i, 0, item); } } availableMasters.sort(); // Sort the masters alphabetically // Now we put the current plugin in the mDataFilesModel under its masters QStandardItem *parent = new QStandardItem(availableMasters.join(",")); QStandardItem *child = new QStandardItem(QString::fromStdString(iter->second.filename().string())); const QList masterList = mDataFilesModel->findItems(availableMasters.join(",")); if (masterList.isEmpty()) { // Masters node not yet in the mDataFilesModel parent->appendRow(child); mDataFilesModel->appendRow(parent); } else { // Masters node exists, append current plugin foreach (QStandardItem *currentItem, masterList) { currentItem->appendRow(child); } } } // TODO: Better dynamic resizing of rows resizeRows(); readConfig(); } void DataFilesPage::setupConfig() { qDebug() << "setupConfig called"; QString config = "./launcher.cfg"; QFile file(config); if (!file.exists()) { config = QString::fromStdString(Files::getPath(Files::Path_ConfigUser, "openmw", "launcher.cfg")); } file.setFileName(config); // Just for displaying information qDebug() << "Using config file from " << file.fileName(); file.open(QIODevice::ReadOnly); qDebug() << "File contents:" << file.readAll(); file.close(); // Open our config file mLauncherConfig = new QSettings(config, QSettings::IniFormat); mLauncherConfig->sync(); // Make sure we have no groups open while (!mLauncherConfig->group().isEmpty()) { mLauncherConfig->endGroup(); } mLauncherConfig->beginGroup("Profiles"); QStringList profiles = mLauncherConfig->childGroups(); if (profiles.isEmpty()) { // Add a default profile profiles.append("Default"); } mProfilesComboBox->addItems(profiles); QString currentProfile = mLauncherConfig->value("CurrentProfile").toString(); qDebug() << mLauncherConfig->value("CurrentProfile").toString(); qDebug() << mLauncherConfig->childGroups(); if (currentProfile.isEmpty()) { qDebug() << "No current profile selected"; currentProfile = "Default"; } mProfilesComboBox->setCurrentIndex(mProfilesComboBox->findText(currentProfile)); mLauncherConfig->endGroup(); // Now we connect the combobox to do something if the profile changes // This prevents strange behaviour while reading and appending the profiles connect(mProfilesComboBox, SIGNAL(textChanged(const QString&, const QString&)), this, SLOT(profileChanged(const QString&, const QString&))); } void DataFilesPage::newProfile() { bool ok; QString text = QInputDialog::getText(this, tr("New Profile"), tr("Profile Name:"), QLineEdit::Normal, tr("New Profile"), &ok); if (ok && !text.isEmpty()) { if (mProfilesComboBox->findText(text) != -1) { QMessageBox::warning(this, tr("Profile already exists"), tr("the profile %0 already exists.").arg(text), QMessageBox::Ok); } else { // Add the new profile to the combobox mProfilesComboBox->addItem(text); mProfilesComboBox->setCurrentIndex(mProfilesComboBox->findText(text)); } } } void DataFilesPage::copyProfile() { QString profile = mProfilesComboBox->currentText(); bool ok; QString text = QInputDialog::getText(this, tr("Copy Profile"), tr("Profile Name:"), QLineEdit::Normal, tr("%0 Copy").arg(profile), &ok); if (ok && !text.isEmpty()) { if (mProfilesComboBox->findText(text) != -1) { QMessageBox::warning(this, tr("Profile already exists"), tr("the profile %0 already exists.").arg(text), QMessageBox::Ok); } else { // Add the new profile to the combobox mProfilesComboBox->addItem(text); // First write the current profile as the new profile writeConfig(text); mProfilesComboBox->setCurrentIndex(mProfilesComboBox->findText(text)); } } } void DataFilesPage::deleteProfile() { QString profile = mProfilesComboBox->currentText(); if (profile.isEmpty()) { return; } QMessageBox deleteMessageBox(this); deleteMessageBox.setWindowTitle(tr("Delete Profile")); deleteMessageBox.setText(tr("Are you sure you want to delete %0?").arg(profile)); deleteMessageBox.setIcon(QMessageBox::Warning); QAbstractButton *deleteButton = deleteMessageBox.addButton(tr("Delete"), QMessageBox::ActionRole); deleteMessageBox.addButton(QMessageBox::Cancel); deleteMessageBox.exec(); if (deleteMessageBox.clickedButton() == deleteButton) { qDebug() << "Delete profile " << profile; // Make sure we have no groups open while (!mLauncherConfig->group().isEmpty()) { mLauncherConfig->endGroup(); } mLauncherConfig->beginGroup("Profiles"); // Open the profile-name subgroup mLauncherConfig->beginGroup(profile); mLauncherConfig->remove(""); // Clear the subgroup mLauncherConfig->endGroup(); mLauncherConfig->endGroup(); // Remove the profile from the combobox mProfilesComboBox->removeItem(mProfilesComboBox->findText(profile)); } } void DataFilesPage::masterSelectionChanged(const QItemSelection &selected, const QItemSelection &deselected) { if (mMastersWidget->selectionModel()->hasSelection()) { QStringList masters; QString masterstr; // Create a QStringList containing all the masters const QStringList masterList = selectedMasters(); foreach (const QString ¤tMaster, masterList) { masters.append(currentMaster); } masters.sort(); masterstr = masters.join(","); // Make a comma-separated QString qDebug() << "Masters" << masterstr; // Iterate over all masters in the datafilesmodel to see if they are selected for (int r=0; rrowCount(); ++r) { QModelIndex currentIndex = mDataFilesModel->index(r, 0); QString master = currentIndex.data().toString(); if (currentIndex.isValid()) { // See if the current master is in the string with selected masters if (masterstr.contains(master)) { // Append the plugins from the current master to pluginsmodel addPlugins(currentIndex); mPluginsTable->resizeRowsToContents(); } } } } // See what plugins to remove QModelIndexList deselectedIndexes = deselected.indexes(); if (!deselectedIndexes.isEmpty()) { foreach (const QModelIndex ¤tIndex, deselectedIndexes) { QString master = currentIndex.data().toString(); master.prepend("*"); master.append("*"); const QList itemList = mDataFilesModel->findItems(master, Qt::MatchWildcard); if (itemList.isEmpty()) qDebug() << "Empty as shit"; foreach (const QStandardItem *currentItem, itemList) { QModelIndex index = currentItem->index(); qDebug() << "Master to remove plugins of:" << index.data().toString(); removePlugins(index); } } } } void DataFilesPage::addPlugins(const QModelIndex &index) { // Find the plugins in the datafilesmodel and append them to the pluginsmodel if (!index.isValid()) return; for (int r=0; rrowCount(index); ++r ) { QModelIndex childIndex = index.child(r, 0); if (childIndex.isValid()) { // Now we see if the pluginsmodel already contains this plugin const QString childIndexData = QVariant(mDataFilesModel->data(childIndex)).toString(); const QList itemList = mPluginsModel->findItems(childIndexData); if (itemList.isEmpty()) { // Plugin not yet in the pluginsmodel, add it QStandardItem *item = new QStandardItem(childIndexData); item->setFlags(item->flags() & ~(Qt::ItemIsDropEnabled)); item->setCheckable(true); mPluginsModel->appendRow(item); } } } } void DataFilesPage::removePlugins(const QModelIndex &index) { if (!index.isValid()) return; for (int r=0; rrowCount(index); ++r) { QModelIndex childIndex = index.child(r, 0); const QList itemList = mPluginsModel->findItems(QVariant(childIndex.data()).toString()); if (!itemList.isEmpty()) { foreach (const QStandardItem *currentItem, itemList) { qDebug() << "Remove plugin:" << currentItem->data(Qt::DisplayRole).toString(); mPluginsModel->removeRow(currentItem->row()); } } } } void DataFilesPage::setCheckstate(QModelIndex index) { if (!index.isValid()) return; QModelIndex sourceModelIndex = mPluginsProxyModel->mapToSource(index); if (mPluginsModel->data(sourceModelIndex, Qt::CheckStateRole) == Qt::Checked) { // Selected row is checked, uncheck it mPluginsModel->setData(sourceModelIndex, Qt::Unchecked, Qt::CheckStateRole); } else { mPluginsModel->setData(sourceModelIndex, Qt::Checked, Qt::CheckStateRole); } } const QStringList DataFilesPage::selectedMasters() { QStringList masters; const QList selectedMasters = mMastersWidget->selectedItems(); foreach (const QTableWidgetItem *item, selectedMasters) { masters.append(item->data(Qt::DisplayRole).toString()); } return masters; } const QStringList DataFilesPage::checkedPlugins() { QStringList checkedItems; for (int r=0; rrowCount(); ++r ) { QModelIndex index = mPluginsModel->index(r, 0); if (index.isValid()) { // See if the current item is checked if (mPluginsModel->data(index, Qt::CheckStateRole) == Qt::Checked) { checkedItems.append(index.data().toString()); } } } return checkedItems; } void DataFilesPage::uncheckPlugins() { for (int r=0; rrowCount(); ++r ) { QModelIndex index = mPluginsModel->index(r, 0); if (index.isValid()) { // See if the current item is checked if (mPluginsModel->data(index, Qt::CheckStateRole) == Qt::Checked) { qDebug() << "Uncheck!"; mPluginsModel->setData(index, Qt::Unchecked, Qt::CheckStateRole); } } } } void DataFilesPage::resizeRows() { // Contents changed mPluginsTable->resizeRowsToContents(); } void DataFilesPage::filterChanged(const QString filter) { QRegExp regExp(filter, Qt::CaseInsensitive, QRegExp::FixedString); mPluginsProxyModel->setFilterRegExp(regExp); resizeRows(); } void DataFilesPage::profileChanged(const QString &previous, const QString ¤t) { qDebug() << "Profile changed " << current << previous; if (!previous.isEmpty()) { writeConfig(previous); mLauncherConfig->sync(); } uncheckPlugins(); // Deselect the masters mMastersWidget->selectionModel()->clearSelection(); readConfig(); } void DataFilesPage::readConfig() { QString profile = mProfilesComboBox->currentText(); qDebug() << "read from: " << profile; // Make sure we have no groups open while (!mLauncherConfig->group().isEmpty()) { mLauncherConfig->endGroup(); } mLauncherConfig->beginGroup("Profiles"); mLauncherConfig->beginGroup(profile); QStringList childKeys = mLauncherConfig->childKeys(); foreach (const QString &key, childKeys) { const QString keyValue = mLauncherConfig->value(key).toString(); if (key.startsWith("Plugin")) { const QList pluginList = mPluginsModel->findItems(keyValue); if (!pluginList.isEmpty()) { foreach (const QStandardItem *currentPlugin, pluginList) { mPluginsModel->setData(currentPlugin->index(), Qt::Checked, Qt::CheckStateRole); } } } if (key.startsWith("Master")) { qDebug() << "Read master: " << keyValue; const QList masterList = mMastersWidget->findItems(keyValue, Qt::MatchFixedString); if (!masterList.isEmpty()) { foreach (QTableWidgetItem *currentMaster, masterList) { mMastersWidget->selectionModel()->select(mMastersWidget->model()->index(currentMaster->row(), 0), QItemSelectionModel::Select); } } } } } void DataFilesPage::writeConfig(QString profile) { // TODO: Testing the config here if (profile.isEmpty()) { profile = mProfilesComboBox->currentText(); } if (profile.isEmpty()) { return; } qDebug() << "Writing: " << profile; // Make sure we have no groups open while (!mLauncherConfig->group().isEmpty()) { mLauncherConfig->endGroup(); } mLauncherConfig->beginGroup("Profiles"); mLauncherConfig->setValue("CurrentProfile", profile); // Open the profile-name subgroup qDebug() << "beginning group: " << profile; mLauncherConfig->beginGroup(profile); mLauncherConfig->remove(""); // Clear the subgroup // First write the masters to the config const QStringList masterList = selectedMasters(); // We don't use foreach because we need i for (int i = 0; i < masterList.size(); ++i) { const QString master = masterList.at(i); mLauncherConfig->setValue(QString("Master%0").arg(i), master); } // Now write all checked plugins const QStringList plugins = checkedPlugins(); for (int i = 0; i < plugins.size(); ++i) { mLauncherConfig->setValue(QString("Plugin%1").arg(i), plugins.at(i)); } mLauncherConfig->endGroup(); mLauncherConfig->endGroup(); }