mirror of
https://github.com/qbittorrent/qBittorrent.git
synced 2025-01-18 13:33:45 +03:00
aa8f7552a2
- Fix crash when renaming currently displayed RSS filter - Remove overwriting confirmation when exporting RSS filters since Qt takes care of it
519 lines
17 KiB
C++
519 lines
17 KiB
C++
/*
|
|
* Bittorrent Client using Qt4 and libtorrent.
|
|
* 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
|
|
*/
|
|
|
|
#ifndef FEEDDOWNLOADER_H
|
|
#define FEEDDOWNLOADER_H
|
|
|
|
#include <QString>
|
|
#include <QSettings>
|
|
#include <QListWidget>
|
|
#include <QListWidgetItem>
|
|
#include <QInputDialog>
|
|
#include <QMessageBox>
|
|
#include <QRegExp>
|
|
#include <QMenu>
|
|
#include <QFile>
|
|
#include <QDataStream>
|
|
#include <QFileDialog>
|
|
|
|
#include "bittorrent.h"
|
|
#include "ui_feeddownloader.h"
|
|
|
|
#ifdef QT_4_5
|
|
#include <QHash>
|
|
#else
|
|
#include <QMap>
|
|
#define QHash QMap
|
|
#define toHash toMap
|
|
#endif
|
|
|
|
class FeedFilter: public QHash<QString, QVariant> {
|
|
private:
|
|
bool valid;
|
|
public:
|
|
FeedFilter():valid(true) {}
|
|
FeedFilter(bool valid): valid(valid) {}
|
|
FeedFilter(QHash<QString, QVariant> filter): QHash<QString, QVariant>(filter), valid(true) {}
|
|
|
|
bool matches(QString s) {
|
|
QStringList match_tokens = getMatchingTokens();
|
|
//qDebug("Checking matching tokens: \"%s\"", getMatchingTokens_str().toLocal8Bit().data());
|
|
foreach(const QString& token, match_tokens) {
|
|
if(token.isEmpty() || token == "")
|
|
continue;
|
|
//qDebug("Token: %s", token.toLocal8Bit().data());
|
|
QRegExp reg(token, Qt::CaseInsensitive, QRegExp::Wildcard);
|
|
//reg.setMinimal(false);
|
|
if(reg.indexIn(s) < 0) return false;
|
|
}
|
|
qDebug("Checking not matching tokens");
|
|
// Checking not matching
|
|
QStringList notmatch_tokens = getNotMatchingTokens();
|
|
foreach(const QString& token, notmatch_tokens) {
|
|
if(token.isEmpty()) continue;
|
|
QRegExp reg(token, Qt::CaseInsensitive, QRegExp::Wildcard);
|
|
if(reg.indexIn(s) > -1) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool isValid() const {
|
|
return valid;
|
|
}
|
|
|
|
QStringList getMatchingTokens() const {
|
|
QString matches = this->value("matches", "").toString();
|
|
return matches.split(" ");
|
|
}
|
|
|
|
QString getMatchingTokens_str() const {
|
|
return this->value("matches", "").toString();
|
|
}
|
|
|
|
void setMatchingTokens(QString tokens) {
|
|
tokens = tokens.trimmed();
|
|
if(tokens.isEmpty())
|
|
(*this)["matches"] = "";
|
|
else
|
|
(*this)["matches"] = tokens;
|
|
}
|
|
|
|
QStringList getNotMatchingTokens() const {
|
|
QString notmatching = this->value("not", "").toString();
|
|
return notmatching.split(" ");
|
|
}
|
|
|
|
QString getNotMatchingTokens_str() const {
|
|
return this->value("not", "").toString();
|
|
}
|
|
|
|
void setNotMatchingTokens(QString tokens) {
|
|
(*this)["not"] = tokens.trimmed();
|
|
}
|
|
|
|
QString getSavePath() const {
|
|
return this->value("save_path", "").toString();
|
|
}
|
|
|
|
void setSavePath(QString save_path) {
|
|
(*this)["save_path"] = save_path;
|
|
}
|
|
|
|
};
|
|
|
|
class FeedFilters : public QHash<QString, QVariant> {
|
|
|
|
private:
|
|
QString feed_url;
|
|
|
|
public:
|
|
FeedFilters() {}
|
|
FeedFilters(QString feed_url, QHash<QString, QVariant> filters): QHash<QString, QVariant>(filters), feed_url(feed_url) {}
|
|
|
|
bool hasFilter(QString name) const {
|
|
return this->contains(name);
|
|
}
|
|
|
|
FeedFilter* matches(QString s) {
|
|
if(!isDownloadingEnabled()) return 0;
|
|
if(this->size() == 0) return new FeedFilter(false);
|
|
foreach(QVariant var_hash_filter, this->values()) {
|
|
QHash<QString, QVariant> hash_filter = var_hash_filter.toHash();
|
|
FeedFilter *filter = new FeedFilter(hash_filter);
|
|
if(filter->matches(s))
|
|
return filter;
|
|
else
|
|
delete filter;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
QStringList names() const {
|
|
return this->keys();
|
|
}
|
|
|
|
FeedFilter getFilter(QString name) const {
|
|
if(this->contains(name))
|
|
return FeedFilter(this->value(name).toHash());
|
|
return FeedFilter();
|
|
}
|
|
|
|
void setFilter(QString name, FeedFilter f) {
|
|
(*this)[name] = f;
|
|
}
|
|
|
|
bool isDownloadingEnabled() const {
|
|
QSettings qBTRSS("qBittorrent", "qBittorrent-rss");
|
|
QHash<QString, QVariant> feeds_w_downloader = qBTRSS.value("downloader_on", QHash<QString, QVariant>()).toHash();
|
|
return feeds_w_downloader.value(feed_url, false).toBool();
|
|
}
|
|
|
|
void setDownloadingEnabled(bool enabled) {
|
|
QSettings qBTRSS("qBittorrent", "qBittorrent-rss");
|
|
QHash<QString, QVariant> feeds_w_downloader = qBTRSS.value("downloader_on", QHash<QString, QVariant>()).toHash();
|
|
feeds_w_downloader[feed_url] = enabled;
|
|
qBTRSS.setValue("downloader_on", feeds_w_downloader);
|
|
}
|
|
|
|
static FeedFilters getFeedFilters(QString url) {
|
|
QSettings qBTRSS("qBittorrent", "qBittorrent-rss");
|
|
QHash<QString, QVariant> all_feeds_filters = qBTRSS.value("feed_filters", QHash<QString, QVariant>()).toHash();
|
|
return FeedFilters(url, all_feeds_filters.value(url, QHash<QString, QVariant>()).toHash());
|
|
}
|
|
|
|
void rename(QString old_name, QString new_name) {
|
|
Q_ASSERT(this->contains(old_name));
|
|
(*this)[new_name] = this->take(old_name);
|
|
}
|
|
|
|
bool serialize(QString path) {
|
|
QFile f(path);
|
|
if(f.open(QIODevice::WriteOnly)) {
|
|
QDataStream out(&f);
|
|
out.setVersion(QDataStream::Qt_4_3);
|
|
out << (QHash<QString, QVariant>)(*this);
|
|
f.close();
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool unserialize(QString path) {
|
|
QFile f(path);
|
|
if(f.open(QIODevice::ReadOnly)) {
|
|
QDataStream in(&f);
|
|
in.setVersion(QDataStream::Qt_4_3);
|
|
QHash<QString, QVariant> tmp;
|
|
in >> tmp;
|
|
qDebug("Unserialized %d filters", tmp.size());
|
|
foreach(const QString& key, tmp.keys()) {
|
|
(*this)[key] = tmp[key];
|
|
}
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void save() {
|
|
QSettings qBTRSS("qBittorrent", "qBittorrent-rss");
|
|
QHash<QString, QVariant> all_feeds_filters = qBTRSS.value("feed_filters", QHash<QString, QVariant>()).toHash();
|
|
qDebug("Saving filters for feed: %s (%d filters)", feed_url.toLocal8Bit().data(), (*this).size());
|
|
all_feeds_filters[feed_url] = *this;
|
|
qBTRSS.setValue("feed_filters", all_feeds_filters);
|
|
}
|
|
};
|
|
|
|
class FeedDownloaderDlg : public QDialog, private Ui_FeedDownloader{
|
|
Q_OBJECT
|
|
|
|
private:
|
|
QString feed_url;
|
|
QString feed_name;
|
|
FeedFilters filters;
|
|
Bittorrent *BTSession;
|
|
QString selected_filter; // name
|
|
|
|
public:
|
|
FeedDownloaderDlg(QWidget *parent, QString feed_url, QString feed_name, Bittorrent* BTSession): QDialog(parent), feed_url(feed_url), feed_name(feed_name), BTSession(BTSession), selected_filter(QString::null){
|
|
setupUi(this);
|
|
setAttribute(Qt::WA_DeleteOnClose);
|
|
Q_ASSERT(!feed_name.isEmpty());
|
|
rssfeed_lbl->setText(feed_name);
|
|
filters = FeedFilters::getFeedFilters(feed_url);
|
|
// Connect Signals/Slots
|
|
connect(filtersList, SIGNAL(currentItemChanged(QListWidgetItem* , QListWidgetItem *)), this, SLOT(showFilterSettings(QListWidgetItem *)));
|
|
connect(filtersList, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(displayFiltersListMenu(const QPoint&)));
|
|
connect(actionAdd_filter, SIGNAL(triggered()), this, SLOT(addFilter()));
|
|
connect(actionRemove_filter, SIGNAL(triggered()), this, SLOT(deleteFilter()));
|
|
connect(actionRename_filter, SIGNAL(triggered()), this, SLOT(renameFilter()));
|
|
connect(del_button, SIGNAL(clicked(bool)), this, SLOT(deleteFilter()));
|
|
connect(add_button, SIGNAL(clicked(bool)), this, SLOT(addFilter()));
|
|
connect(enableDl_cb, SIGNAL(stateChanged(int)), this, SLOT(enableFilterBox(int)));
|
|
// Restore saved info
|
|
enableDl_cb->setChecked(filters.isDownloadingEnabled());
|
|
fillFiltersList();
|
|
filtersList->sortItems(Qt::AscendingOrder);
|
|
if(filters.size() > 0) {
|
|
// Select first filter
|
|
filtersList->setCurrentItem(filtersList->item(0));
|
|
//showFilterSettings(filtersList->item(0));
|
|
}
|
|
// Show
|
|
show();
|
|
}
|
|
|
|
~FeedDownloaderDlg() {
|
|
if(enableDl_cb->isChecked())
|
|
emit filteringEnabled();
|
|
// Make sure we save everything
|
|
saveCurrentFilterSettings();
|
|
filters.save();
|
|
}
|
|
|
|
protected slots:
|
|
void saveCurrentFilterSettings() {
|
|
if(!selected_filter.isEmpty()) {
|
|
FeedFilter filter = filters.getFilter(selected_filter);
|
|
filter.setMatchingTokens(match_line->text());
|
|
filter.setNotMatchingTokens(notmatch_line->text());
|
|
QString save_path = savepath_line->text().trimmed();
|
|
if(save_path.isEmpty())
|
|
save_path = BTSession->getDefaultSavePath();
|
|
filter.setSavePath(save_path);
|
|
// Save updated filter
|
|
filters.setFilter(selected_filter, filter);
|
|
}
|
|
}
|
|
|
|
void on_browse_button_clicked() {
|
|
QString default_path = savepath_line->text();
|
|
if(default_path.isEmpty() || !QDir(default_path).exists()) {
|
|
default_path = QDir::homePath();
|
|
}
|
|
QString dir = QFileDialog::getExistingDirectory(this, tr("Choose save path"), QDir::homePath());
|
|
if(!dir.isNull() and QDir(dir).exists()) {
|
|
savepath_line->setText(dir);
|
|
}
|
|
}
|
|
|
|
void fillFiltersList() {
|
|
// Fill filter list
|
|
foreach(QString filter_name, filters.names()) {
|
|
new QListWidgetItem(filter_name, filtersList);
|
|
}
|
|
}
|
|
|
|
void displayFiltersListMenu(const QPoint&) {
|
|
QMenu myFiltersListMenu(this);
|
|
if(filtersList->selectedItems().size() > 0) {
|
|
myFiltersListMenu.addAction(actionRename_filter);
|
|
myFiltersListMenu.addAction(actionRemove_filter);
|
|
} else {
|
|
myFiltersListMenu.addAction(actionAdd_filter);
|
|
}
|
|
// Call menu
|
|
myFiltersListMenu.exec(QCursor::pos());
|
|
}
|
|
|
|
void showFilterSettings(QListWidgetItem *item) {
|
|
// First, save current filter settings
|
|
saveCurrentFilterSettings();
|
|
// Clear all fields
|
|
clearFields();
|
|
if(!item) {
|
|
qDebug("No new selected item");
|
|
return;
|
|
}
|
|
// Actually show filter settings
|
|
QString filter_name = item->text();
|
|
FeedFilter filter = filters.getFilter(filter_name);
|
|
filterSettingsBox->setEnabled(true);
|
|
match_line->setText(filter.getMatchingTokens_str());
|
|
if(match_line->text().trimmed().isEmpty()) {
|
|
match_line->setText(filter_name);
|
|
}
|
|
notmatch_line->setText(filter.getNotMatchingTokens_str());
|
|
QString save_path = filter.getSavePath();
|
|
if(save_path.isEmpty())
|
|
save_path = BTSession->getDefaultSavePath();
|
|
savepath_line->setText(save_path);
|
|
// Update selected filter
|
|
selected_filter = filter_name;
|
|
}
|
|
|
|
void deleteFilter() {
|
|
QList<QListWidgetItem *> items = filtersList->selectedItems();
|
|
if(items.size() == 1) {
|
|
QListWidgetItem * item = items.first();
|
|
filters.remove(item->text());
|
|
selected_filter = QString::null;
|
|
delete item;
|
|
// Reset Filter settings view
|
|
if(filters.size() == 0) {
|
|
clearFields();
|
|
filterSettingsBox->setEnabled(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
void renameFilter() {
|
|
QList<QListWidgetItem *> items = filtersList->selectedItems();
|
|
if(items.size() == 1) {
|
|
QListWidgetItem *item = items.first();
|
|
QString current_name = item->text();
|
|
QString new_name;
|
|
bool validated = false;
|
|
do {
|
|
new_name = askFilterName(current_name);
|
|
if(new_name.isNull() || new_name == current_name) return;
|
|
if(!filters.hasFilter(new_name)) {
|
|
validated = true;
|
|
} else {
|
|
QMessageBox::warning(0, tr("Invalid filter name"), tr("This filter name is already in use."));
|
|
}
|
|
}while(!validated);
|
|
// Save the current filter
|
|
saveCurrentFilterSettings();
|
|
// Rename the filter
|
|
filters.rename(current_name, new_name);
|
|
if(selected_filter == current_name)
|
|
selected_filter = new_name;
|
|
item->setText(new_name);
|
|
filtersList->sortItems(Qt::AscendingOrder);
|
|
}
|
|
}
|
|
|
|
void enableFilterBox(int state) {
|
|
if(state == Qt::Checked) {
|
|
filtersBox->setEnabled(true);
|
|
filters.setDownloadingEnabled(true);
|
|
} else {
|
|
filtersBox->setEnabled(false);
|
|
filters.setDownloadingEnabled(false);
|
|
}
|
|
}
|
|
|
|
QString askFilterName(QString name=QString::null) {
|
|
QString name_prop;
|
|
if(name.isEmpty())
|
|
name_prop = tr("New filter");
|
|
else
|
|
name_prop = name;
|
|
QString new_name;
|
|
bool validated = false;
|
|
do {
|
|
bool ok;
|
|
new_name = QInputDialog::getText(this, tr("Please choose a name for this filter"), tr("Filter name:"), QLineEdit::Normal, name_prop, &ok);
|
|
if(!ok) {
|
|
return QString::null;
|
|
}
|
|
// Validate filter name
|
|
new_name = new_name.trimmed();
|
|
if(new_name.isEmpty()) {
|
|
// Cannot be left empty
|
|
QMessageBox::warning(0, tr("Invalid filter name"), tr("The filter name cannot be left empty."));
|
|
} else {
|
|
validated = true;
|
|
}
|
|
} while(!validated);
|
|
return new_name;
|
|
}
|
|
|
|
void addFilter() {
|
|
QString filter_name = QString::null;
|
|
bool validated = false;
|
|
do {
|
|
filter_name = askFilterName();
|
|
if(filter_name.isNull()) return;
|
|
if(filters.hasFilter(filter_name)) {
|
|
// Filter alread exists
|
|
QMessageBox::warning(0, tr("Invalid filter name"), tr("This filter name is already in use."));
|
|
} else {
|
|
validated = true;
|
|
}
|
|
}while(!validated);
|
|
QListWidgetItem *it = new QListWidgetItem(filter_name, filtersList);
|
|
filtersList->sortItems(Qt::AscendingOrder);
|
|
filtersList->setCurrentItem(it);
|
|
//showFilterSettings(it);
|
|
}
|
|
|
|
void clearFields() {
|
|
match_line->clear();
|
|
notmatch_line->clear();
|
|
savepath_line->clear();
|
|
test_res_lbl->setText("");
|
|
test_line->clear();
|
|
}
|
|
|
|
void on_testButton_clicked(bool) {
|
|
test_res_lbl->clear();
|
|
if(selected_filter.isEmpty()) {
|
|
qDebug("No filter is selected!!!");
|
|
return;
|
|
}
|
|
QString s = test_line->text().trimmed();
|
|
if(s.isEmpty()) {
|
|
QMessageBox::warning(0, tr("Filter testing error"), tr("Please specify a test torrent name."));
|
|
return;
|
|
}
|
|
// Get current filter
|
|
saveCurrentFilterSettings();
|
|
FeedFilter f = filters.getFilter(selected_filter);
|
|
if(f.matches(s))
|
|
test_res_lbl->setText("<b><font color=\"green\">"+tr("matches")+"</font></b>");
|
|
else
|
|
test_res_lbl->setText("<b><font color=\"red\">"+tr("does not match")+"</font></b>");
|
|
}
|
|
|
|
void on_importButton_clicked(bool) {
|
|
QString source = QFileDialog::getOpenFileName(0, tr("Select file to import"), QDir::homePath(), tr("Filters Files")+QString::fromUtf8(" (*.filters)"));
|
|
if(source.isEmpty()) return;
|
|
if(filters.unserialize(source)) {
|
|
// Clean up first
|
|
clearFields();
|
|
filtersList->clear();
|
|
selected_filter = QString::null;
|
|
fillFiltersList();
|
|
if(filters.size() > 0)
|
|
filtersList->setCurrentItem(filtersList->item(0));
|
|
QMessageBox::information(0, tr("Import successful"), tr("Filters import was successful."));
|
|
} else {
|
|
QMessageBox::warning(0, tr("Import failure"), tr("Filters could not be imported due to an I/O error."));
|
|
}
|
|
}
|
|
|
|
void on_exportButton_clicked(bool) {
|
|
QString destination = QFileDialog::getSaveFileName(this, tr("Select destination file"), QDir::homePath(), tr("Filters Files")+QString::fromUtf8(" (*.filters)"));
|
|
if(destination.isEmpty()) return;
|
|
// Append file extension
|
|
if(!destination.endsWith(".filters"))
|
|
destination += ".filters";
|
|
/*if(QFile::exists(destination)) {
|
|
int ret = QMessageBox::question(0, tr("Overwriting confirmation"), tr("Are you sure you want to overwrite existing file?"), QMessageBox::Yes|QMessageBox::No);
|
|
if(ret != QMessageBox::Yes) return;
|
|
}*/
|
|
if(filters.serialize(destination))
|
|
QMessageBox::information(0, tr("Export successful"), tr("Filters export was successful."));
|
|
else
|
|
QMessageBox::warning(0, tr("Export failure"), tr("Filters could not be exported due to an I/O error."));
|
|
}
|
|
|
|
signals:
|
|
void filteringEnabled();
|
|
|
|
};
|
|
|
|
#undef QHash
|
|
#undef toHash
|
|
|
|
#endif // FEEDDOWNLOADER_H
|