/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2015 Vladimir Golovnev * Copyright (C) 2006 Christophe Dumez * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * In addition, as a special exception, the copyright holders give permission to * link this program with the OpenSSL project's "OpenSSL" library (or with * modified versions of it that use the same license as the "OpenSSL" library), * and distribute the linked executables. You must obey the GNU General Public * License in all respects for all of the code used other than "OpenSSL". If you * modify file(s), you may extend this exception to your version of the file(s), * but you are not obligated to do so. If you do not wish to do so, delete this * exception statement from your version. * * Contact : chris@qbittorrent.org */ #include "pluginselectdlg.h" #include #include #include #include #include #include #include #include #include #include "autoexpandabledialog.h" #include "base/net/downloadhandler.h" #include "base/net/downloadmanager.h" #include "base/utils/fs.h" #include "base/utils/misc.h" #include "guiiconprovider.h" #include "pluginsourcedlg.h" #include "searchwidget.h" #include "ui_pluginselectdlg.h" #include "utils.h" enum PluginColumns { PLUGIN_NAME, PLUGIN_VERSION, PLUGIN_URL, PLUGIN_STATE, PLUGIN_ID }; PluginSelectDlg::PluginSelectDlg(SearchPluginManager *pluginManager, QWidget *parent) : QDialog(parent) , m_ui(new Ui::PluginSelectDlg()) , m_pluginManager(pluginManager) , m_asyncOps(0) , m_pendingUpdates(0) { m_ui->setupUi(this); setAttribute(Qt::WA_DeleteOnClose); // This hack fixes reordering of first column with Qt5. // https://github.com/qtproject/qtbase/commit/e0fc088c0c8bc61dbcaf5928b24986cd61a22777 QTableView unused; unused.setVerticalHeader(m_ui->pluginsTree->header()); m_ui->pluginsTree->header()->setParent(m_ui->pluginsTree); unused.setVerticalHeader(new QHeaderView(Qt::Horizontal)); m_ui->pluginsTree->setRootIsDecorated(false); m_ui->pluginsTree->hideColumn(PLUGIN_ID); m_ui->pluginsTree->header()->setSortIndicator(0, Qt::AscendingOrder); m_ui->actionUninstall->setIcon(GuiIconProvider::instance()->getIcon("list-remove")); connect(m_ui->actionEnable, &QAction::toggled, this, &PluginSelectDlg::enableSelection); connect(m_ui->pluginsTree, &QTreeWidget::customContextMenuRequested, this, &PluginSelectDlg::displayContextMenu); connect(m_ui->pluginsTree, &QTreeWidget::itemDoubleClicked, this, &PluginSelectDlg::togglePluginState); loadSupportedSearchPlugins(); connect(m_pluginManager, &SearchPluginManager::pluginInstalled, this, &PluginSelectDlg::pluginInstalled); connect(m_pluginManager, &SearchPluginManager::pluginInstallationFailed, this, &PluginSelectDlg::pluginInstallationFailed); connect(m_pluginManager, &SearchPluginManager::pluginUpdated, this, &PluginSelectDlg::pluginUpdated); connect(m_pluginManager, &SearchPluginManager::pluginUpdateFailed, this, &PluginSelectDlg::pluginUpdateFailed); connect(m_pluginManager, &SearchPluginManager::checkForUpdatesFinished, this, &PluginSelectDlg::checkForUpdatesFinished); connect(m_pluginManager, &SearchPluginManager::checkForUpdatesFailed, this, &PluginSelectDlg::checkForUpdatesFailed); Utils::Gui::resize(this); show(); } PluginSelectDlg::~PluginSelectDlg() { delete m_ui; } void PluginSelectDlg::dropEvent(QDropEvent *event) { event->acceptProposedAction(); QStringList files; if (event->mimeData()->hasUrls()) { foreach (const QUrl &url, event->mimeData()->urls()) { if (!url.isEmpty()) { if (url.scheme().compare("file", Qt::CaseInsensitive) == 0) files << url.toLocalFile(); else files << url.toString(); } } } else { files = event->mimeData()->text().split(QLatin1String("\n")); } if (files.isEmpty()) return; foreach (QString file, files) { qDebug("dropped %s", qUtf8Printable(file)); startAsyncOp(); m_pluginManager->installPlugin(file); } } // Decode if we accept drag 'n drop or not void PluginSelectDlg::dragEnterEvent(QDragEnterEvent *event) { QString mime; foreach (mime, event->mimeData()->formats()) { qDebug("mimeData: %s", qUtf8Printable(mime)); } if (event->mimeData()->hasFormat(QLatin1String("text/plain")) || event->mimeData()->hasFormat(QLatin1String("text/uri-list"))) { event->acceptProposedAction(); } } void PluginSelectDlg::on_updateButton_clicked() { startAsyncOp(); m_pluginManager->checkForUpdates(); } void PluginSelectDlg::togglePluginState(QTreeWidgetItem *item, int) { PluginInfo *plugin = m_pluginManager->pluginInfo(item->text(PLUGIN_ID)); m_pluginManager->enablePlugin(plugin->name, !plugin->enabled); if (plugin->enabled) { item->setText(PLUGIN_STATE, tr("Yes")); setRowColor(m_ui->pluginsTree->indexOfTopLevelItem(item), "green"); } else { item->setText(PLUGIN_STATE, tr("No")); setRowColor(m_ui->pluginsTree->indexOfTopLevelItem(item), "red"); } } void PluginSelectDlg::displayContextMenu(const QPoint&) { QMenu myContextMenu(this); // Enable/disable pause/start action given the DL state QList items = m_ui->pluginsTree->selectedItems(); if (items.isEmpty()) return; QString first_id = items.first()->text(PLUGIN_ID); m_ui->actionEnable->setChecked(m_pluginManager->pluginInfo(first_id)->enabled); myContextMenu.addAction(m_ui->actionEnable); myContextMenu.addSeparator(); myContextMenu.addAction(m_ui->actionUninstall); myContextMenu.exec(QCursor::pos()); } void PluginSelectDlg::on_closeButton_clicked() { close(); } void PluginSelectDlg::on_actionUninstall_triggered() { bool error = false; foreach (QTreeWidgetItem *item, m_ui->pluginsTree->selectedItems()) { int index = m_ui->pluginsTree->indexOfTopLevelItem(item); Q_ASSERT(index != -1); QString id = item->text(PLUGIN_ID); if (m_pluginManager->uninstallPlugin(id)) { delete item; } else { error = true; // Disable it instead m_pluginManager->enablePlugin(id, false); item->setText(PLUGIN_STATE, tr("No")); setRowColor(index, "red"); } } if (error) QMessageBox::warning(this, tr("Uninstall warning"), tr("Some plugins could not be uninstalled because they are included in qBittorrent. Only the ones you added yourself can be uninstalled.\nThose plugins were disabled.")); else QMessageBox::information(this, tr("Uninstall success"), tr("All selected plugins were uninstalled successfully")); } void PluginSelectDlg::enableSelection(bool enable) { foreach (QTreeWidgetItem *item, m_ui->pluginsTree->selectedItems()) { int index = m_ui->pluginsTree->indexOfTopLevelItem(item); Q_ASSERT(index != -1); QString id = item->text(PLUGIN_ID); m_pluginManager->enablePlugin(id, enable); if (enable) { item->setText(PLUGIN_STATE, tr("Yes")); setRowColor(index, "green"); } else { item->setText(PLUGIN_STATE, tr("No")); setRowColor(index, "red"); } } } // Set the color of a row in data model void PluginSelectDlg::setRowColor(int row, QString color) { QTreeWidgetItem *item = m_ui->pluginsTree->topLevelItem(row); for (int i = 0; i < m_ui->pluginsTree->columnCount(); ++i) { item->setData(i, Qt::ForegroundRole, QVariant(QColor(color))); } } QList PluginSelectDlg::findItemsWithUrl(QString url) { QList res; for (int i = 0; i < m_ui->pluginsTree->topLevelItemCount(); ++i) { QTreeWidgetItem *item = m_ui->pluginsTree->topLevelItem(i); if (url.startsWith(item->text(PLUGIN_URL), Qt::CaseInsensitive)) res << item; } return res; } QTreeWidgetItem* PluginSelectDlg::findItemWithID(QString id) { for (int i = 0; i < m_ui->pluginsTree->topLevelItemCount(); ++i) { QTreeWidgetItem *item = m_ui->pluginsTree->topLevelItem(i); if (id == item->text(PLUGIN_ID)) return item; } return 0; } void PluginSelectDlg::loadSupportedSearchPlugins() { // Some clean up first m_ui->pluginsTree->clear(); foreach (QString name, m_pluginManager->allPlugins()) addNewPlugin(name); } void PluginSelectDlg::addNewPlugin(QString pluginName) { QTreeWidgetItem *item = new QTreeWidgetItem(m_ui->pluginsTree); PluginInfo *plugin = m_pluginManager->pluginInfo(pluginName); item->setText(PLUGIN_NAME, plugin->fullName); item->setText(PLUGIN_URL, plugin->url); item->setText(PLUGIN_ID, plugin->name); if (plugin->enabled) { item->setText(PLUGIN_STATE, tr("Yes")); setRowColor(m_ui->pluginsTree->indexOfTopLevelItem(item), "green"); } else { item->setText(PLUGIN_STATE, tr("No")); setRowColor(m_ui->pluginsTree->indexOfTopLevelItem(item), "red"); } // Handle icon if (QFile::exists(plugin->iconPath)) { // Good, we already have the icon item->setData(PLUGIN_NAME, Qt::DecorationRole, QVariant(QIcon(plugin->iconPath))); } else { // Icon is missing, we must download it using namespace Net; DownloadHandler *handler = DownloadManager::instance()->downloadUrl(plugin->url + "/favicon.ico", true); connect(handler, static_cast(&DownloadHandler::downloadFinished) , this, &PluginSelectDlg::iconDownloaded); connect(handler, &DownloadHandler::downloadFailed, this, &PluginSelectDlg::iconDownloadFailed); } item->setText(PLUGIN_VERSION, plugin->version); } void PluginSelectDlg::startAsyncOp() { ++m_asyncOps; if (m_asyncOps == 1) setCursor(QCursor(Qt::WaitCursor)); } void PluginSelectDlg::finishAsyncOp() { --m_asyncOps; if (m_asyncOps == 0) setCursor(QCursor(Qt::ArrowCursor)); } void PluginSelectDlg::finishPluginUpdate() { --m_pendingUpdates; if (m_pendingUpdates == 0 && !m_updatedPlugins.isEmpty()) { m_updatedPlugins.sort(Qt::CaseInsensitive); QMessageBox::information(this, tr("Search plugin update"), tr("Plugins installed or updated: %1").arg(m_updatedPlugins.join(", "))); m_updatedPlugins.clear(); } } void PluginSelectDlg::on_installButton_clicked() { PluginSourceDlg *dlg = new PluginSourceDlg(this); connect(dlg, &PluginSourceDlg::askForLocalFile, this, &PluginSelectDlg::askForLocalPlugin); connect(dlg, &PluginSourceDlg::askForUrl, this, &PluginSelectDlg::askForPluginUrl); } void PluginSelectDlg::askForPluginUrl() { bool ok = false; QString clipTxt = qApp->clipboard()->text(); QString defaultUrl = "http://"; if (Utils::Misc::isUrl(clipTxt) && clipTxt.endsWith(".py")) defaultUrl = clipTxt; QString url = AutoExpandableDialog::getText( this, tr("New search engine plugin URL"), tr("URL:"), QLineEdit::Normal, defaultUrl, &ok ); while (ok && !url.isEmpty() && !url.endsWith(".py")) { QMessageBox::warning(this, tr("Invalid link"), tr("The link doesn't seem to point to a search engine plugin.")); url = AutoExpandableDialog::getText( this, tr("New search engine plugin URL"), tr("URL:"), QLineEdit::Normal, url, &ok ); } if (ok && !url.isEmpty()) { startAsyncOp(); m_pluginManager->installPlugin(url); } } void PluginSelectDlg::askForLocalPlugin() { QStringList pathsList = QFileDialog::getOpenFileNames( 0, tr("Select search plugins"), QDir::homePath(), tr("qBittorrent search plugin") + QLatin1String(" (*.py)") ); foreach (QString path, pathsList) { startAsyncOp(); m_pluginManager->installPlugin(path); } } void PluginSelectDlg::iconDownloaded(const QString &url, QString filePath) { filePath = Utils::Fs::fromNativePath(filePath); // Icon downloaded QIcon icon(filePath); // Detect a non-decodable icon QList sizes = icon.availableSizes(); bool invalid = (sizes.isEmpty() || icon.pixmap(sizes.first()).isNull()); if (!invalid) { foreach (QTreeWidgetItem *item, findItemsWithUrl(url)) { QString id = item->text(PLUGIN_ID); PluginInfo *plugin = m_pluginManager->pluginInfo(id); if (!plugin) continue; QString iconPath = QString("%1/%2.%3") .arg(SearchPluginManager::pluginsLocation()) .arg(id) .arg(url.endsWith(".ico", Qt::CaseInsensitive) ? "ico" : "png"); if (QFile::copy(filePath, iconPath)) { // This 2nd check is necessary. Some favicons (eg from piratebay) // decode fine without an ext, but fail to do so when appending the ext // from the url. Probably a Qt bug. QIcon iconWithExt(iconPath); QList sizesExt = iconWithExt.availableSizes(); bool invalidExt = (sizesExt.isEmpty() || iconWithExt.pixmap(sizesExt.first()).isNull()); if (invalidExt) { Utils::Fs::forceRemove(iconPath); continue; } item->setData(PLUGIN_NAME, Qt::DecorationRole, iconWithExt); m_pluginManager->updateIconPath(plugin); } } } // Delete tmp file Utils::Fs::forceRemove(filePath); } void PluginSelectDlg::iconDownloadFailed(const QString &url, const QString &reason) { qDebug("Could not download favicon: %s, reason: %s", qUtf8Printable(url), qUtf8Printable(reason)); } void PluginSelectDlg::checkForUpdatesFinished(const QHash &updateInfo) { finishAsyncOp(); if (updateInfo.isEmpty()) { QMessageBox::information(this, tr("Search plugin update"), tr("All your plugins are already up to date.")); return; } for (auto i = updateInfo.cbegin(); i != updateInfo.cend(); ++i) { startAsyncOp(); m_pendingUpdates++; m_pluginManager->updatePlugin(i.key()); } } void PluginSelectDlg::checkForUpdatesFailed(const QString &reason) { finishAsyncOp(); QMessageBox::warning(this, tr("Search plugin update"), tr("Sorry, couldn't check for plugin updates. %1").arg(reason)); } void PluginSelectDlg::pluginInstalled(const QString &name) { addNewPlugin(name); finishAsyncOp(); m_updatedPlugins.append(name); finishPluginUpdate(); } void PluginSelectDlg::pluginInstallationFailed(const QString &name, const QString &reason) { finishAsyncOp(); QMessageBox::information(this, tr("Search plugin install"), tr("Couldn't install \"%1\" search engine plugin. %2").arg(name).arg(reason)); finishPluginUpdate(); } void PluginSelectDlg::pluginUpdated(const QString &name) { finishAsyncOp(); PluginVersion version = m_pluginManager->pluginInfo(name)->version; QTreeWidgetItem *item = findItemWithID(name); item->setText(PLUGIN_VERSION, version); m_updatedPlugins.append(name); finishPluginUpdate(); } void PluginSelectDlg::pluginUpdateFailed(const QString &name, const QString &reason) { finishAsyncOp(); QMessageBox::information(this, tr("Search plugin update"), tr("Couldn't update \"%1\" search engine plugin. %2").arg(name).arg(reason)); finishPluginUpdate(); }