1
0
mirror of https://github.com/d47081/qBittorrent.git synced 2025-01-10 23:07:59 +00:00

Refactor side panel code. Encapsulate each widget's logic in their own subclass.

This commit is contained in:
sledgehammer999 2015-03-26 03:39:45 +02:00
parent 1bb968f9fd
commit ff1aaa8733
3 changed files with 468 additions and 473 deletions

View File

@ -1583,16 +1583,18 @@ void Preferences::setTorrentLabels(const QStringList& labels)
void Preferences::addTorrentLabel(const QString& label)
{
QStringList labels = value("TransferListFilters/customLabels").toStringList();
if (!labels.contains(label))
labels << label;
if (labels.contains(label))
return;
labels << label;
setValue("TransferListFilters/customLabels", labels);
}
void Preferences::removeTorrentLabel(const QString& label)
{
QStringList labels = value("TransferListFilters/customLabels").toStringList();
if (labels.contains(label))
labels.removeOne(label);
if (!labels.contains(label))
return;
labels.removeOne(label);
setValue("TransferListFilters/customLabels", labels);
}

View File

@ -51,13 +51,21 @@
#include "downloadthread.h"
#include "logger.h"
FiltersBase::FiltersBase(QWidget *parent)
FiltersBase::FiltersBase(QWidget *parent, TransferListWidget *transferList)
: QListWidget(parent)
, transferList(transferList)
{
setStyleSheet("QListWidget { background: transparent; border: 0 }");
#if defined(Q_OS_MAC)
setAttribute(Qt::WA_MacShowFocusRect, false);
#endif
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setContextMenuPolicy(Qt::CustomContextMenu);
connect(this, SIGNAL(customContextMenuRequested(QPoint)), SLOT(showMenu(QPoint)));
connect(this, SIGNAL(currentRowChanged(int)), SLOT(applyFilter(int)));
connect(transferList->getSourceModel(), SIGNAL(torrentAdded(TorrentModelItem*)), SLOT(handleNewTorrent(TorrentModelItem*)));
connect(transferList->getSourceModel(), SIGNAL(torrentAboutToBeRemoved(TorrentModelItem*)), SLOT(torrentAboutToBeDeleted(TorrentModelItem*)));
}
QSize FiltersBase::sizeHint() const
@ -76,26 +84,298 @@ QSize FiltersBase::minimumSizeHint() const
return size;
}
LabelFiltersList::LabelFiltersList(QWidget *parent)
: FiltersBase(parent)
StatusFiltersWidget::StatusFiltersWidget(QWidget *parent, TransferListWidget *transferList)
: FiltersBase(parent, transferList)
{
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
setUniformItemSizes(true);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
// Height is fixed (sizeHint().height() is used)
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
setSpacing(0);
connect(transferList->getSourceModel(), SIGNAL(modelRefreshed()), SLOT(updateTorrentNumbers()));
// Add status filters
QListWidgetItem *all = new QListWidgetItem(this);
all->setData(Qt::DisplayRole, QVariant(tr("All") + " (0)"));
all->setData(Qt::DecorationRole, QIcon(":/icons/skin/filterall.png"));
QListWidgetItem *downloading = new QListWidgetItem(this);
downloading->setData(Qt::DisplayRole, QVariant(tr("Downloading") + " (0)"));
downloading->setData(Qt::DecorationRole, QIcon(":/icons/skin/downloading.png"));
QListWidgetItem *completed = new QListWidgetItem(this);
completed->setData(Qt::DisplayRole, QVariant(tr("Completed") + " (0)"));
completed->setData(Qt::DecorationRole, QIcon(":/icons/skin/uploading.png"));
QListWidgetItem *resumed = new QListWidgetItem(this);
resumed->setData(Qt::DisplayRole, QVariant(tr("Resumed") + " (0)"));
resumed->setData(Qt::DecorationRole, QIcon(":/icons/skin/resumed.png"));
QListWidgetItem *paused = new QListWidgetItem(this);
paused->setData(Qt::DisplayRole, QVariant(tr("Paused") + " (0)"));
paused->setData(Qt::DecorationRole, QIcon(":/icons/skin/paused.png"));
QListWidgetItem *active = new QListWidgetItem(this);
active->setData(Qt::DisplayRole, QVariant(tr("Active") + " (0)"));
active->setData(Qt::DecorationRole, QIcon(":/icons/skin/filteractive.png"));
QListWidgetItem *inactive = new QListWidgetItem(this);
inactive->setData(Qt::DisplayRole, QVariant(tr("Inactive") + " (0)"));
inactive->setData(Qt::DecorationRole, QIcon(":/icons/skin/filterinactive.png"));
setCurrentRow(Preferences::instance()->getTransSelFilter(), QItemSelectionModel::SelectCurrent);
}
void LabelFiltersList::addItem(QListWidgetItem *it)
StatusFiltersWidget::~StatusFiltersWidget()
{
Preferences::instance()->setTransSelFilter(currentRow());
}
void StatusFiltersWidget::updateTorrentNumbers()
{
const TorrentStatusReport report = transferList->getSourceModel()->getTorrentStatusReport();
item(TorrentFilter::ALL)->setData(Qt::DisplayRole, QVariant(tr("All (%1)").arg(report.nb_active + report.nb_inactive)));
item(TorrentFilter::DOWNLOADING)->setData(Qt::DisplayRole, QVariant(tr("Downloading (%1)").arg(report.nb_downloading)));
item(TorrentFilter::COMPLETED)->setData(Qt::DisplayRole, QVariant(tr("Completed (%1)").arg(report.nb_seeding)));
item(TorrentFilter::PAUSED)->setData(Qt::DisplayRole, QVariant(tr("Paused (%1)").arg(report.nb_paused)));
item(TorrentFilter::RESUMED)->setData(Qt::DisplayRole, QVariant(tr("Resumed (%1)").arg(report.nb_active + report.nb_inactive - report.nb_paused)));
item(TorrentFilter::ACTIVE)->setData(Qt::DisplayRole, QVariant(tr("Active (%1)").arg(report.nb_active)));
item(TorrentFilter::INACTIVE)->setData(Qt::DisplayRole, QVariant(tr("Inactive (%1)").arg(report.nb_inactive)));
}
void StatusFiltersWidget::showMenu(QPoint) {}
void StatusFiltersWidget::applyFilter(int row)
{
transferList->applyStatusFilter(row);
}
void StatusFiltersWidget::handleNewTorrent(TorrentModelItem*) {}
void StatusFiltersWidget::torrentAboutToBeDeleted(TorrentModelItem*) {}
LabelFiltersList::LabelFiltersList(QWidget *parent, TransferListWidget *transferList)
: FiltersBase(parent, transferList)
, m_totalTorrents(0)
, m_totalLabeled(0)
{
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
connect(transferList->getSourceModel(), SIGNAL(torrentChangedLabel(TorrentModelItem*,QString,QString)), SLOT(torrentChangedLabel(TorrentModelItem*, QString, QString)));
// Add Label filters
QListWidgetItem *allLabels = new QListWidgetItem(this);
allLabels->setData(Qt::DisplayRole, QVariant(tr("All (0)", "this is for the label filter")));
allLabels->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
QListWidgetItem *noLabel = new QListWidgetItem(this);
noLabel->setData(Qt::DisplayRole, QVariant(tr("Unlabeled (0)")));
noLabel->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
QStringList labelList = Preferences::instance()->getTorrentLabels();
for (int i=0; i < labelList.size(); ++i)
addItem(labelList[i], false);
setCurrentRow(0, QItemSelectionModel::SelectCurrent);
}
LabelFiltersList::~LabelFiltersList()
{
Preferences::instance()->setTorrentLabels(m_labels.keys());
}
void LabelFiltersList::addItem(QString &label, bool hasTorrent)
{
int labelCount = 0;
QListWidgetItem *labelItem = 0;
label = fsutils::toValidFileSystemName(label.trimmed());
item(0)->setText(tr("All (%1)", "this is for the label filter").arg(m_totalTorrents));
if (label.isEmpty()) {
item(1)->setText(tr("Unlabeled (%1)").arg(m_totalTorrents - m_totalLabeled));
return;
}
bool exists = m_labels.contains(label);
if (exists) {
labelCount = m_labels.value(label);
labelItem = item(rowFromLabel(label));
}
else {
labelItem = new QListWidgetItem();
labelItem->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
}
if (hasTorrent) {
++m_totalLabeled;
++labelCount;
}
item(1)->setText(tr("Unlabeled (%1)").arg(m_totalTorrents - m_totalLabeled));
Preferences::instance()->addTorrentLabel(label);
m_labels.insert(label, labelCount);
labelItem->setText(tr("%1 (%2)", "label_name (10)").arg(label).arg(labelCount));
if (exists)
return;
Q_ASSERT(count() >= 2);
for (int i = 2; i<count(); ++i) {
if (item(i)->text().localeAwareCompare(it->text()) >= 0) {
insertItem(i, it);
bool less = false;
if (!(misc::naturalSort(label, item(i)->text(), less)))
less = (label.localeAwareCompare(item(i)->text()) < 0);
if (less) {
insertItem(i, labelItem);
updateGeometry();
return;
}
}
QListWidget::addItem(it);
QListWidget::addItem(labelItem);
updateGeometry();
}
void LabelFiltersList::removeItem(const QString &label)
{
item(0)->setText(tr("All (%1)", "this is for the label filter").arg(m_totalTorrents));
if (label.isEmpty()) {
// In case we here from torrentAboutToBeDeleted()
item(1)->setText(tr("Unlabeled (%1)").arg(m_totalTorrents - m_totalLabeled));
return;
}
--m_totalLabeled;
item(1)->setText(tr("Unlabeled (%1)").arg(m_totalTorrents - m_totalLabeled));
int labelCount = m_labels.value(label) - 1;
int row = rowFromLabel(label);
if (row < 2)
return;
QListWidgetItem *labelItem = item(row);
labelItem->setText(tr("%1 (%2)", "label_name (10)").arg(label).arg(labelCount));
m_labels.insert(label, labelCount);
}
void LabelFiltersList::removeSelectedLabel()
{
const int labelRow = row(selectedItems().first());
if (labelRow < 2)
return;
const QString &label = labelFromRow(labelRow);
Q_ASSERT(m_labels.contains(label));
m_labels.remove(label);
// Select first label
setCurrentRow(0, QItemSelectionModel::SelectCurrent);
applyFilter(0);
// Un display filter
delete takeItem(labelRow);
transferList->removeLabelFromRows(label);
// Save custom labels to remember it was deleted
Preferences::instance()->removeTorrentLabel(label);
updateGeometry();
}
void LabelFiltersList::removeUnusedLabels()
{
QStringList unusedLabels;
QHash<QString, int>::const_iterator i;
for (i = m_labels.begin(); i != m_labels.end(); ++i) {
if (i.value() == 0)
unusedLabels << i.key();
}
foreach (const QString &label, unusedLabels) {
m_labels.remove(label);
delete takeItem(rowFromLabel(label));
Preferences::instance()->removeTorrentLabel(label);
}
if (!unusedLabels.isEmpty())
updateGeometry();
}
void LabelFiltersList::torrentChangedLabel(TorrentModelItem *torrentItem, QString old_label, QString new_label)
{
Q_UNUSED(torrentItem);
qDebug("Torrent label changed from %s to %s", qPrintable(old_label), qPrintable(new_label));
removeItem(old_label);
addItem(new_label, true);
}
void LabelFiltersList::showMenu(QPoint)
{
QMenu menu(this);
QAction *addAct = menu.addAction(IconProvider::instance()->getIcon("list-add"), tr("Add label..."));
QAction *removeAct = 0;
QAction *removeUnusedAct = 0;
if (!selectedItems().empty() && row(selectedItems().first()) > 1)
removeAct = menu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove label"));
removeUnusedAct = menu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove unused labels"));
menu.addSeparator();
QAction *startAct = menu.addAction(IconProvider::instance()->getIcon("media-playback-start"), tr("Resume torrents"));
QAction *pauseAct = menu.addAction(IconProvider::instance()->getIcon("media-playback-pause"), tr("Pause torrents"));
QAction *deleteTorrentsAct = menu.addAction(IconProvider::instance()->getIcon("edit-delete"), tr("Delete torrents"));
QAction *act = 0;
act = menu.exec(QCursor::pos());
if (!act)
return;
if (act == removeAct) {
removeSelectedLabel();
}
else if (act == removeUnusedAct) {
removeUnusedLabels();
}
else if (act == deleteTorrentsAct) {
transferList->deleteVisibleTorrents();
}
else if (act == startAct) {
transferList->startVisibleTorrents();
}
else if (act == pauseAct) {
transferList->pauseVisibleTorrents();
}
else if (act == addAct) {
bool ok;
QString label = "";
bool invalid;
do {
invalid = false;
label = AutoExpandableDialog::getText(this, tr("New Label"), tr("Label:"), QLineEdit::Normal, label, &ok);
if (ok && !label.isEmpty()) {
if (fsutils::isValidFileSystemName(label)) {
addItem(label, false);
}
else {
QMessageBox::warning(this, tr("Invalid label name"), tr("Please don't use any special characters in the label name."));
invalid = true;
}
}
} while (invalid);
}
}
void LabelFiltersList::applyFilter(int row)
{
switch (row) {
case 0:
transferList->applyLabelFilterAll();
break;
case 1:
transferList->applyLabelFilter(QString());
break;
default:
transferList->applyLabelFilter(labelFromRow(row));
}
}
void LabelFiltersList::handleNewTorrent(TorrentModelItem* torrentItem)
{
++m_totalTorrents;
QString label = torrentItem->data(TorrentModelItem::TR_LABEL).toString();
addItem(label, true);
// labelFilters->addItem() may have changed the label, update the model accordingly.
torrentItem->setData(TorrentModelItem::TR_LABEL, label);
}
void LabelFiltersList::torrentAboutToBeDeleted(TorrentModelItem* torrentItem)
{
--m_totalTorrents;
Q_ASSERT(torrentItem);
QString label = torrentItem->data(TorrentModelItem::TR_LABEL).toString();
removeItem(label);
}
QString LabelFiltersList::labelFromRow(int row) const
{
Q_ASSERT(row > 1);
@ -106,7 +386,7 @@ QString LabelFiltersList::labelFromRow(int row) const
return parts.join(" ");
}
int LabelFiltersList::rowFromLabel(QString label) const
int LabelFiltersList::rowFromLabel(const QString &label) const
{
Q_ASSERT(!label.isEmpty());
for (int i = 2; i<count(); ++i)
@ -114,27 +394,22 @@ int LabelFiltersList::rowFromLabel(QString label) const
return -1;
}
StatusFiltersWidget::StatusFiltersWidget(QWidget *parent)
: FiltersBase(parent)
{
setUniformItemSizes(true);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
// Height is fixed (sizeHint().height() is used)
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
}
TrackerFiltersList::TrackerFiltersList(QWidget *parent)
: FiltersBase(parent)
TrackerFiltersList::TrackerFiltersList(QWidget *parent, TransferListWidget *transferList)
: FiltersBase(parent, transferList)
, m_downloader(new DownloadThread(this))
, m_totalTorrents(0)
{
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
QListWidgetItem *allTrackers = new QListWidgetItem(this);
allTrackers->setData(Qt::DisplayRole, QVariant(tr("All trackers (0)")));
allTrackers->setData(Qt::DisplayRole, QVariant(tr("All (0)", "this is for the label filter")));
allTrackers->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("network-server"));
QListWidgetItem *noTracker = new QListWidgetItem(this);
noTracker->setData(Qt::DisplayRole, QVariant(tr("Trackerless (0)")));
noTracker->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("network-server"));
m_trackers.insert("", QStringList());
setCurrentRow(0, QItemSelectionModel::SelectCurrent);
connect(m_downloader, SIGNAL(downloadFinished(QString, QString)), SLOT(handleFavicoDownload(QString, QString)));
connect(m_downloader, SIGNAL(downloadFailure(QString, QString)), SLOT(handleFavicoFailure(QString, QString)));
}
@ -151,8 +426,9 @@ void TrackerFiltersList::addItem(const QString &tracker, const QString &hash)
QStringList tmp;
QListWidgetItem *trackerItem = 0;
QString host = getHost(tracker);
bool exists = m_trackers.contains(host);
if (m_trackers.contains(host)) {
if (exists) {
tmp = m_trackers.value(host);
if (tmp.contains(hash))
return;
@ -178,6 +454,11 @@ void TrackerFiltersList::addItem(const QString &tracker, const QString &hash)
}
trackerItem->setText(tr("%1 (%2)", "openbittorrent.com (10)").arg(host).arg(tmp.size()));
if (exists) {
if (currentRow() == rowFromTracker(host))
emit currentRowChanged(currentRow());
return;
}
Q_ASSERT(count() >= 2);
for (int i = 2; i<count(); ++i) {
@ -186,24 +467,12 @@ void TrackerFiltersList::addItem(const QString &tracker, const QString &hash)
less = (host.localeAwareCompare(item(i)->text()) < 0);
if (less) {
insertItem(i, trackerItem);
if (currentRow() == i)
emit currentRowChanged(i);
updateGeometry();
return;
}
}
QListWidget::addItem(trackerItem);
}
void TrackerFiltersList::addItem(const QTorrentHandle& handle)
{
QString hash = handle.hash();
std::vector<libtorrent::announce_entry> trackers = handle.trackers();
for (std::vector<libtorrent::announce_entry>::iterator i = trackers.begin(), e = trackers.end(); i != e; ++i)
addItem(misc::toQStringU(i->url), hash);
//Check for trackerless torrent
if (trackers.size() == 0)
addItem("", hash);
updateGeometry();
}
void TrackerFiltersList::removeItem(const QString &tracker, const QString &hash)
@ -222,9 +491,10 @@ void TrackerFiltersList::removeItem(const QString &tracker, const QString &hash)
trackerItem = item(row);
if (tmp.empty()) {
if (currentRow() == row)
setCurrentRow(0);
setCurrentRow(0, QItemSelectionModel::SelectCurrent);
delete trackerItem;
m_trackers.remove(host);
updateGeometry();
return;
}
trackerItem->setText(tr("%1 (%2)", "openbittorrent.com (10)").arg(host).arg(tmp.size()));
@ -240,29 +510,12 @@ void TrackerFiltersList::removeItem(const QString &tracker, const QString &hash)
emit currentRowChanged(row);
}
void TrackerFiltersList::removeItem(const QTorrentHandle& handle)
void TrackerFiltersList::changeTrackerless(bool trackerless, const QString &hash)
{
QString hash = handle.hash();
std::vector<libtorrent::announce_entry> trackers = handle.trackers();
for (std::vector<libtorrent::announce_entry>::iterator i = trackers.begin(), e = trackers.end(); i != e; ++i)
removeItem(misc::toQStringU(i->url), hash);
//Check for trackerless torrent
if (trackers.size() == 0)
removeItem("", hash);
}
void TrackerFiltersList::setTorrentCount(int all)
{
item(0)->setText(tr("All trackers (%1)").arg(all));
}
QStringList TrackerFiltersList::getHashes(int row)
{
if (row == 1)
return m_trackers.value("");
if (trackerless)
addItem("", hash);
else
return m_trackers.value(trackerFromRow(row));
removeItem("", hash);
}
void TrackerFiltersList::handleFavicoDownload(const QString& url, const QString& filePath)
@ -300,12 +553,62 @@ void TrackerFiltersList::handleFavicoFailure(const QString& url, const QString&
Log::WARNING);
}
void TrackerFiltersList::changeTrackerless(bool trackerless, const QString &hash)
void TrackerFiltersList::showMenu(QPoint)
{
if (trackerless)
addItem("", hash);
QMenu menu(this);
QAction *startAct = menu.addAction(IconProvider::instance()->getIcon("media-playback-start"), tr("Resume torrents"));
QAction *pauseAct = menu.addAction(IconProvider::instance()->getIcon("media-playback-pause"), tr("Pause torrents"));
QAction *deleteTorrentsAct = menu.addAction(IconProvider::instance()->getIcon("edit-delete"), tr("Delete torrents"));
QAction *act = 0;
act = menu.exec(QCursor::pos());
if (!act)
return;
if (act == startAct)
transferList->startVisibleTorrents();
else if (act == pauseAct)
transferList->pauseVisibleTorrents();
else if (act == deleteTorrentsAct)
transferList->deleteVisibleTorrents();
}
void TrackerFiltersList::applyFilter(int row)
{
if (row == 0)
transferList->applyTrackerFilterAll();
else
transferList->applyTrackerFilter(getHashes(row));
}
void TrackerFiltersList::handleNewTorrent(TorrentModelItem* torrentItem)
{
QTorrentHandle handle = torrentItem->torrentHandle();
QString hash = handle.hash();
std::vector<libtorrent::announce_entry> trackers = handle.trackers();
for (std::vector<libtorrent::announce_entry>::iterator i = trackers.begin(), e = trackers.end(); i != e; ++i)
addItem(misc::toQStringU(i->url), hash);
//Check for trackerless torrent
if (trackers.size() == 0)
addItem("", hash);
item(0)->setText(tr("All (%1)", "this is for the tracker filter").arg(++m_totalTorrents));
}
void TrackerFiltersList::torrentAboutToBeDeleted(TorrentModelItem* torrentItem)
{
QTorrentHandle handle = torrentItem->torrentHandle();
QString hash = handle.hash();
std::vector<libtorrent::announce_entry> trackers = handle.trackers();
for (std::vector<libtorrent::announce_entry>::iterator i = trackers.begin(), e = trackers.end(); i != e; ++i)
removeItem(misc::toQStringU(i->url), hash);
//Check for trackerless torrent
if (trackers.size() == 0)
removeItem("", hash);
item(0)->setText(tr("All (%1)", "this is for the tracker filter").arg(--m_totalTorrents));
}
QString TrackerFiltersList::trackerFromRow(int row) const
@ -338,102 +641,47 @@ QString TrackerFiltersList::getHost(const QString &trakcer) const
return longHost.mid(index + 1);
}
TransferListFiltersWidget::TransferListFiltersWidget(QWidget *parent, TransferListWidget *transferList): QFrame(parent), transferList(transferList), nb_labeled(0), nb_torrents(0)
QStringList TrackerFiltersList::getHashes(int row)
{
if (row == 1)
return m_trackers.value("");
else
return m_trackers.value(trackerFromRow(row));
}
TransferListFiltersWidget::TransferListFiltersWidget(QWidget *parent, TransferListWidget *transferList)
: QFrame(parent)
, statusFilters(0)
, trackerFilters(0)
, torrentsLabel(0)
{
// Construct lists
vLayout = new QVBoxLayout();
QVBoxLayout *vLayout = new QVBoxLayout(this);
vLayout->setContentsMargins(0, 4, 0, 0);
QFont font;
font.setBold(true);
font.setCapitalization(QFont::SmallCaps);
torrentsLabel = new QLabel(tr("Torrents"));
torrentsLabel = new QLabel(tr("Torrents"), this);
torrentsLabel->setIndent(2);
torrentsLabel->setFont(font);
vLayout->addWidget(torrentsLabel);
statusFilters = new StatusFiltersWidget(this);
statusFilters = new StatusFiltersWidget(this, transferList);
vLayout->addWidget(statusFilters);
QLabel *labelsLabel = new QLabel(tr("Labels"));
QLabel *labelsLabel = new QLabel(tr("Labels"), this);
labelsLabel->setIndent(2);
labelsLabel->setFont(font);
vLayout->addWidget(labelsLabel);
labelFilters = new LabelFiltersList(this);
LabelFiltersList *labelFilters = new LabelFiltersList(this, transferList);
vLayout->addWidget(labelFilters);
QLabel *trackersLabel = new QLabel(tr("Trackers"));
QLabel *trackersLabel = new QLabel(tr("Trackers"), this);
trackersLabel->setIndent(2);
trackersLabel->setFont(font);
vLayout->addWidget(trackersLabel);
trackerFilters = new TrackerFiltersList(this);
trackerFilters = new TrackerFiltersList(this, transferList);
vLayout->addWidget(trackerFilters);
setLayout(vLayout);
labelFilters->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
trackerFilters->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
statusFilters->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
statusFilters->setSpacing(0);
setContentsMargins(0,0,0,0);
vLayout->setSpacing(2);
// Add status filters
QListWidgetItem *all = new QListWidgetItem(statusFilters);
all->setData(Qt::DisplayRole, QVariant(tr("All") + " (0)"));
all->setData(Qt::DecorationRole, QIcon(":/icons/skin/filterall.png"));
QListWidgetItem *downloading = new QListWidgetItem(statusFilters);
downloading->setData(Qt::DisplayRole, QVariant(tr("Downloading") + " (0)"));
downloading->setData(Qt::DecorationRole, QIcon(":/icons/skin/downloading.png"));
QListWidgetItem *completed = new QListWidgetItem(statusFilters);
completed->setData(Qt::DisplayRole, QVariant(tr("Completed") + " (0)"));
completed->setData(Qt::DecorationRole, QIcon(":/icons/skin/uploading.png"));
QListWidgetItem *resumed = new QListWidgetItem(statusFilters);
resumed->setData(Qt::DisplayRole, QVariant(tr("Resumed") + " (0)"));
resumed->setData(Qt::DecorationRole, QIcon(":/icons/skin/resumed.png"));
QListWidgetItem *paused = new QListWidgetItem(statusFilters);
paused->setData(Qt::DisplayRole, QVariant(tr("Paused") + " (0)"));
paused->setData(Qt::DecorationRole, QIcon(":/icons/skin/paused.png"));
QListWidgetItem *active = new QListWidgetItem(statusFilters);
active->setData(Qt::DisplayRole, QVariant(tr("Active") + " (0)"));
active->setData(Qt::DecorationRole, QIcon(":/icons/skin/filteractive.png"));
QListWidgetItem *inactive = new QListWidgetItem(statusFilters);
inactive->setData(Qt::DisplayRole, QVariant(tr("Inactive") + " (0)"));
inactive->setData(Qt::DecorationRole, QIcon(":/icons/skin/filterinactive.png"));
// SIGNAL/SLOT
connect(statusFilters, SIGNAL(currentRowChanged(int)), transferList, SLOT(applyStatusFilter(int)));
connect(transferList->getSourceModel(), SIGNAL(modelRefreshed()), SLOT(updateTorrentNumbers()));
connect(transferList->getSourceModel(), SIGNAL(torrentAdded(TorrentModelItem*)), SLOT(handleNewTorrent(TorrentModelItem*)));
connect(labelFilters, SIGNAL(currentRowChanged(int)), this, SLOT(applyLabelFilter(int)));
connect(trackerFilters, SIGNAL(currentRowChanged(int)), this, SLOT(applyTrackerFilter(int)));
connect(transferList->getSourceModel(), SIGNAL(torrentAboutToBeRemoved(TorrentModelItem*)), SLOT(torrentAboutToBeDeleted(TorrentModelItem*)));
connect(transferList->getSourceModel(), SIGNAL(torrentChangedLabel(TorrentModelItem*,QString,QString)), SLOT(torrentChangedLabel(TorrentModelItem*, QString, QString)));
// Add Label filters
QListWidgetItem *allLabels = new QListWidgetItem(labelFilters);
allLabels->setData(Qt::DisplayRole, QVariant(tr("All labels") + " (0)"));
allLabels->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
QListWidgetItem *noLabel = new QListWidgetItem(labelFilters);
noLabel->setData(Qt::DisplayRole, QVariant(tr("Unlabeled") + " (0)"));
noLabel->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
// Load settings
loadSettings();
labelFilters->setCurrentRow(0);
trackerFilters->setCurrentRow(0);
//labelFilters->selectionModel()->select(labelFilters->model()->index(0,0), QItemSelectionModel::Select);
// Label menu
labelFilters->setContextMenuPolicy(Qt::CustomContextMenu);
connect(labelFilters, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(showLabelMenu(QPoint)));
// Tracker menu
trackerFilters->setContextMenuPolicy(Qt::CustomContextMenu);
connect(trackerFilters, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(showTrackerMenu(QPoint)));
}
TransferListFiltersWidget::~TransferListFiltersWidget()
{
saveSettings();
delete statusFilters;
delete labelFilters;
delete trackerFilters;
delete vLayout;
}
void TransferListFiltersWidget::resizeEvent(QResizeEvent *event)
@ -443,56 +691,6 @@ void TransferListFiltersWidget::resizeEvent(QResizeEvent *event)
trackerFilters->setMinimumHeight((height - minHeight) / 2);
}
StatusFiltersWidget* TransferListFiltersWidget::getStatusFilters() const
{
return statusFilters;
}
void TransferListFiltersWidget::saveSettings() const
{
Preferences* const pref = Preferences::instance();
pref->setTransSelFilter(statusFilters->currentRow());
pref->setTorrentLabels(customLabels.keys());
}
void TransferListFiltersWidget::loadSettings()
{
statusFilters->setCurrentRow(Preferences::instance()->getTransSelFilter());
const QStringList label_list = Preferences::instance()->getTorrentLabels();
foreach (const QString &label, label_list) {
customLabels.insert(label, 0);
qDebug("Creating label QListWidgetItem: %s", qPrintable(label));
QListWidgetItem *newLabel = new QListWidgetItem();
newLabel->setText(label + " (0)");
newLabel->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
labelFilters->addItem(newLabel);
}
}
void TransferListFiltersWidget::updateTorrentNumbers()
{
const TorrentStatusReport report = transferList->getSourceModel()->getTorrentStatusReport();
statusFilters->item(TorrentFilter::ALL)->setData(Qt::DisplayRole, QVariant(tr("All") + " (" + QString::number(report.nb_active + report.nb_inactive) + ")"));
statusFilters->item(TorrentFilter::DOWNLOADING)->setData(Qt::DisplayRole, QVariant(tr("Downloading") + " (" + QString::number(report.nb_downloading) + ")"));
statusFilters->item(TorrentFilter::COMPLETED)->setData(Qt::DisplayRole, QVariant(tr("Completed") + " (" + QString::number(report.nb_seeding) + ")"));
statusFilters->item(TorrentFilter::PAUSED)->setData(Qt::DisplayRole, QVariant(tr("Paused") + " (" + QString::number(report.nb_paused) + ")"));
statusFilters->item(TorrentFilter::RESUMED)->setData(Qt::DisplayRole, QVariant(tr("Resumed") + " (" + QString::number(report.nb_active + report.nb_inactive - report.nb_paused) + ")"));
statusFilters->item(TorrentFilter::ACTIVE)->setData(Qt::DisplayRole, QVariant(tr("Active") + " (" + QString::number(report.nb_active) + ")"));
statusFilters->item(TorrentFilter::INACTIVE)->setData(Qt::DisplayRole, QVariant(tr("Inactive") + " (" + QString::number(report.nb_inactive) + ")"));
}
void TransferListFiltersWidget::addLabel(QString& label)
{
label = fsutils::toValidFileSystemName(label.trimmed());
if (label.isEmpty() || customLabels.contains(label)) return;
QListWidgetItem *newLabel = new QListWidgetItem();
newLabel->setText(label + " (0)");
newLabel->setData(Qt::DecorationRole, IconProvider::instance()->getIcon("inode-directory"));
labelFilters->addItem(newLabel);
customLabels.insert(label, 0);
Preferences::instance()->addTorrentLabel(label);
}
void TransferListFiltersWidget::addTrackers(const QStringList &trackers, const QString &hash)
{
foreach (const QString &tracker, trackers)
@ -509,226 +707,3 @@ void TransferListFiltersWidget::changeTrackerless(bool trackerless, const QStrin
{
trackerFilters->changeTrackerless(trackerless, hash);
}
void TransferListFiltersWidget::showLabelMenu(QPoint)
{
QMenu labelMenu(labelFilters);
QAction *addAct = labelMenu.addAction(IconProvider::instance()->getIcon("list-add"), tr("Add label..."));
QAction *removeAct = 0;
QAction *removeUnusedAct = 0;
if (!labelFilters->selectedItems().empty() && labelFilters->row(labelFilters->selectedItems().first()) > 1)
removeAct = labelMenu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove label"));
else
removeUnusedAct = labelMenu.addAction(IconProvider::instance()->getIcon("list-remove"), tr("Remove unused labels"));
labelMenu.addSeparator();
QAction *startAct = labelMenu.addAction(IconProvider::instance()->getIcon("media-playback-start"), tr("Resume torrents"));
QAction *pauseAct = labelMenu.addAction(IconProvider::instance()->getIcon("media-playback-pause"), tr("Pause torrents"));
QAction *deleteTorrentsAct = labelMenu.addAction(IconProvider::instance()->getIcon("edit-delete"), tr("Delete torrents"));
QAction *act = 0;
act = labelMenu.exec(QCursor::pos());
if (act) {
if (act == removeAct) {
removeSelectedLabel();
return;
}
if (act == removeUnusedAct) {
removeUnusedLabels();
return;
}
if (act == deleteTorrentsAct) {
transferList->deleteVisibleTorrents();
return;
}
if (act == startAct) {
transferList->startVisibleTorrents();
return;
}
if (act == pauseAct) {
transferList->pauseVisibleTorrents();
return;
}
if (act == addAct) {
bool ok;
QString label = "";
bool invalid;
do {
invalid = false;
label = AutoExpandableDialog::getText(this, tr("New Label"), tr("Label:"), QLineEdit::Normal, label, &ok);
if (ok && !label.isEmpty()) {
if (fsutils::isValidFileSystemName(label)) {
addLabel(label);
}
else {
QMessageBox::warning(this, tr("Invalid label name"), tr("Please don't use any special characters in the label name."));
invalid = true;
}
}
} while (invalid);
return;
}
}
}
void TransferListFiltersWidget::showTrackerMenu(QPoint)
{
QMenu trackerMenu(trackerFilters);
QAction *startAct = trackerMenu.addAction(IconProvider::instance()->getIcon("media-playback-start"), tr("Resume torrents"));
QAction *pauseAct = trackerMenu.addAction(IconProvider::instance()->getIcon("media-playback-pause"), tr("Pause torrents"));
QAction *deleteTorrentsAct = trackerMenu.addAction(IconProvider::instance()->getIcon("edit-delete"), tr("Delete torrents"));
QAction *act = 0;
act = trackerMenu.exec(QCursor::pos());
if (act) {
if (act == startAct)
transferList->startVisibleTorrents();
else if (act == pauseAct)
transferList->pauseVisibleTorrents();
else if (act == deleteTorrentsAct)
transferList->deleteVisibleTorrents();
}
}
void TransferListFiltersWidget::removeSelectedLabel()
{
const int row = labelFilters->row(labelFilters->selectedItems().first());
Q_ASSERT(row > 1);
const QString &label = labelFilters->labelFromRow(row);
Q_ASSERT(customLabels.contains(label));
customLabels.remove(label);
transferList->removeLabelFromRows(label);
// Select first label
labelFilters->setCurrentItem(labelFilters->item(0));
labelFilters->selectionModel()->select(labelFilters->model()->index(0,0), QItemSelectionModel::Select);
applyLabelFilter(0);
// Un display filter
delete labelFilters->takeItem(row);
labelFilters->updateGeometry();
// Save custom labels to remember it was deleted
Preferences::instance()->removeTorrentLabel(label);
}
void TransferListFiltersWidget::removeUnusedLabels()
{
QStringList unusedLabels;
QHash<QString, int>::const_iterator i;
for (i = customLabels.begin(); i != customLabels.end(); ++i)
if (i.value() == 0)
unusedLabels << i.key();
foreach (const QString &label, unusedLabels) {
customLabels.remove(label);
delete labelFilters->takeItem(labelFilters->rowFromLabel(label));
Preferences::instance()->removeTorrentLabel(label);
}
if (!unusedLabels.empty())
labelFilters->updateGeometry();
}
void TransferListFiltersWidget::applyLabelFilter(int row)
{
switch (row) {
case 0:
transferList->applyLabelFilterAll();
break;
case 1:
transferList->applyLabelFilter(QString());
break;
default:
transferList->applyLabelFilter(labelFilters->labelFromRow(row));
}
}
void TransferListFiltersWidget::applyTrackerFilter(int row)
{
if (row == 0)
transferList->applyTrackerFilterAll();
else
transferList->applyTrackerFilter(trackerFilters->getHashes(row));
}
void TransferListFiltersWidget::torrentChangedLabel(TorrentModelItem *torrentItem, QString old_label, QString new_label)
{
Q_UNUSED(torrentItem);
qDebug("Torrent label changed from %s to %s", qPrintable(old_label), qPrintable(new_label));
if (!old_label.isEmpty()) {
if (customLabels.contains(old_label)) {
const int new_count = customLabels.value(old_label, 0) - 1;
Q_ASSERT(new_count >= 0);
customLabels.insert(old_label, new_count);
const int row = labelFilters->rowFromLabel(old_label);
Q_ASSERT(row >= 2);
labelFilters->item(row)->setText(old_label + " (" + QString::number(new_count) + ")");
}
--nb_labeled;
}
if (!new_label.isEmpty()) {
if (!customLabels.contains(new_label))
addLabel(new_label);
const int new_count = customLabels.value(new_label, 0) + 1;
Q_ASSERT(new_count >= 1);
customLabels.insert(new_label, new_count);
const int row = labelFilters->rowFromLabel(new_label);
Q_ASSERT(row >= 2);
labelFilters->item(row)->setText(new_label + " (" + QString::number(new_count) + ")");
++nb_labeled;
}
updateStickyLabelCounters();
}
void TransferListFiltersWidget::handleNewTorrent(TorrentModelItem* torrentItem)
{
QString label = torrentItem->data(TorrentModelItem::TR_LABEL).toString();
qDebug("New torrent was added with label: %s", qPrintable(label));
if (!label.isEmpty()) {
if (!customLabels.contains(label)) {
addLabel(label);
// addLabel may have changed the label, update the model accordingly.
torrentItem->setData(TorrentModelItem::TR_LABEL, label);
}
// Update label counter
Q_ASSERT(customLabels.contains(label));
const int new_count = customLabels.value(label, 0) + 1;
customLabels.insert(label, new_count);
const int row = labelFilters->rowFromLabel(label);
qDebug("torrentAdded, Row: %d", row);
Q_ASSERT(row >= 2);
Q_ASSERT(labelFilters->item(row));
labelFilters->item(row)->setText(label + " (" + QString::number(new_count) + ")");
++nb_labeled;
}
++nb_torrents;
Q_ASSERT(nb_torrents >= 0);
Q_ASSERT(nb_labeled >= 0);
Q_ASSERT(nb_labeled <= nb_torrents);
updateStickyLabelCounters();
trackerFilters->addItem(torrentItem->torrentHandle());
trackerFilters->setTorrentCount(nb_torrents);
}
void TransferListFiltersWidget::torrentAboutToBeDeleted(TorrentModelItem* torrentItem)
{
Q_ASSERT(torrentItem);
QString label = torrentItem->data(TorrentModelItem::TR_LABEL).toString();
if (!label.isEmpty()) {
// Update label counter
const int new_count = customLabels.value(label, 0) - 1;
customLabels.insert(label, new_count);
const int row = labelFilters->rowFromLabel(label);
Q_ASSERT(row >= 2);
labelFilters->item(row)->setText(label + " (" + QString::number(new_count) + ")");
--nb_labeled;
}
--nb_torrents;
qDebug("nb_torrents: %d, nb_labeled: %d", nb_torrents, nb_labeled);
Q_ASSERT(nb_torrents >= 0);
Q_ASSERT(nb_labeled >= 0);
Q_ASSERT(nb_labeled <= nb_torrents);
updateStickyLabelCounters();
trackerFilters->removeItem(torrentItem->torrentHandle());
trackerFilters->setTorrentCount(nb_torrents);
}
void TransferListFiltersWidget::updateStickyLabelCounters()
{
labelFilters->item(0)->setText(tr("All labels") + " (" + QString::number(nb_torrents) + ")");
labelFilters->item(1)->setText(tr("Unlabeled") + " (" + QString::number(nb_torrents - nb_labeled) + ")");
}

View File

@ -35,9 +35,7 @@
#include <QFrame>
QT_BEGIN_NAMESPACE
class QListWidgetItem;
class QVBoxLayout;
class QDragMoveEvent;
class QResizeEvent;
class QLabel;
QT_END_NAMESPACE
@ -51,24 +49,19 @@ class FiltersBase: public QListWidget
Q_OBJECT
public:
FiltersBase(QWidget *parent);
FiltersBase(QWidget *parent, TransferListWidget *transferList);
QSize sizeHint() const;
QSize minimumSizeHint() const;
};
virtual QSize sizeHint() const;
virtual QSize minimumSizeHint() const;
class LabelFiltersList: public FiltersBase
{
Q_OBJECT
protected:
TransferListWidget *transferList;
public:
LabelFiltersList(QWidget *parent);
// Redefine addItem() to make sure the list stays sorted
void addItem(QListWidgetItem *it);
QString labelFromRow(int row) const;
int rowFromLabel(QString label) const;
private slots:
virtual void showMenu(QPoint) = 0;
virtual void applyFilter(int row) = 0;
virtual void handleNewTorrent(TorrentModelItem* torrentItem) = 0;
virtual void torrentAboutToBeDeleted(TorrentModelItem* torrentItem) = 0;
};
class StatusFiltersWidget: public FiltersBase
@ -76,7 +69,52 @@ class StatusFiltersWidget: public FiltersBase
Q_OBJECT
public:
StatusFiltersWidget(QWidget *parent);
StatusFiltersWidget(QWidget *parent, TransferListWidget *transferList);
~StatusFiltersWidget();
private slots:
void updateTorrentNumbers();
private:
// These 4 methods are virtual slots in the base class.
// No need to redeclare them here as slots.
virtual void showMenu(QPoint);
virtual void applyFilter(int row);
virtual void handleNewTorrent(TorrentModelItem*);
virtual void torrentAboutToBeDeleted(TorrentModelItem*);
};
class LabelFiltersList: public FiltersBase
{
Q_OBJECT
public:
LabelFiltersList(QWidget *parent, TransferListWidget *transferList);
~LabelFiltersList();
private slots:
// Redefine addItem() to make sure the list stays sorted
void addItem(QString &label, bool hasTorrent);
void removeItem(const QString &label);
void removeSelectedLabel();
void removeUnusedLabels();
void torrentChangedLabel(TorrentModelItem *torrentItem, QString old_label, QString new_label);
private:
// These 4 methods are virtual slots in the base class.
// No need to redeclare them here as slots.
virtual void showMenu(QPoint);
virtual void applyFilter(int row);
virtual void handleNewTorrent(TorrentModelItem* torrentItem);
virtual void torrentAboutToBeDeleted(TorrentModelItem* torrentItem);
QString labelFromRow(int row) const;
int rowFromLabel(const QString &label) const;
private:
QHash<QString, int> m_labels;
int m_totalTorrents;
int m_totalLabeled;
};
class TrackerFiltersList: public FiltersBase
@ -84,55 +122,43 @@ class TrackerFiltersList: public FiltersBase
Q_OBJECT
public:
TrackerFiltersList(QWidget *parent);
TrackerFiltersList(QWidget *parent, TransferListWidget *transferList);
~TrackerFiltersList();
// Redefine addItem() to make sure the list stays sorted
void addItem(const QString &tracker, const QString &hash);
void addItem(const QTorrentHandle &handle);
void removeItem(const QString &tracker, const QString &hash);
void removeItem(const QTorrentHandle &handle);
void setTorrentCount(int all);
QStringList getHashes(int row);
public slots:
void handleFavicoDownload(const QString &url, const QString &filePath);
void handleFavicoFailure(const QString &url, const QString &reason);
void changeTrackerless(bool trackerless, const QString &hash);
private slots:
void handleFavicoDownload(const QString &url, const QString &filePath);
void handleFavicoFailure(const QString &url, const QString &reason);
private:
QHash<QString, QStringList> m_trackers;
// These 4 methods are virtual slots in the base class.
// No need to redeclare them here as slots.
virtual void showMenu(QPoint);
virtual void applyFilter(int row);
virtual void handleNewTorrent(TorrentModelItem* torrentItem);
virtual void torrentAboutToBeDeleted(TorrentModelItem* torrentItem);
QString trackerFromRow(int row) const;
int rowFromTracker(const QString &tracker) const;
QString getHost(const QString &trakcer) const;
QStringList getHashes(int row);
private:
QHash<QString, QStringList> m_trackers;
DownloadThread *m_downloader;
QStringList m_iconPaths;
int m_totalTorrents;
};
class TransferListFiltersWidget: public QFrame
{
Q_OBJECT
private:
QHash<QString, int> customLabels;
StatusFiltersWidget* statusFilters;
LabelFiltersList* labelFilters;
TrackerFiltersList* trackerFilters;
QVBoxLayout* vLayout;
TransferListWidget *transferList;
int nb_labeled;
int nb_torrents;
//for use in resizeEvent()
QLabel *torrentsLabel;
public:
TransferListFiltersWidget(QWidget *parent, TransferListWidget *transferList);
~TransferListFiltersWidget();
StatusFiltersWidget* getStatusFilters() const;
void saveSettings() const;
void loadSettings();
public slots:
void addTrackers(const QStringList &trackers, const QString &hash);
@ -142,19 +168,11 @@ public slots:
protected:
virtual void resizeEvent(QResizeEvent *event);
protected slots:
void updateTorrentNumbers();
void addLabel(QString& label);
void showLabelMenu(QPoint);
void showTrackerMenu(QPoint);
void removeSelectedLabel();
void removeUnusedLabels();
void applyLabelFilter(int row);
void applyTrackerFilter(int row);
void torrentChangedLabel(TorrentModelItem *torrentItem, QString old_label, QString new_label);
void handleNewTorrent(TorrentModelItem* torrentItem);
void torrentAboutToBeDeleted(TorrentModelItem* torrentItem);
void updateStickyLabelCounters();
private:
StatusFiltersWidget *statusFilters;
TrackerFiltersList *trackerFilters;
//for use in resizeEvent()
QLabel *torrentsLabel;
};
#endif // TRANSFERLISTFILTERSWIDGET_H