mirror of
https://github.com/qbittorrent/qBittorrent
synced 2025-08-19 12:59:56 -07:00
Merge pull request #5613 from glassez/cat_tree
Implement category filter widget. Closes #5444.
This commit is contained in:
commit
cb0f73da57
8 changed files with 912 additions and 327 deletions
442
src/gui/categoryfiltermodel.cpp
Normal file
442
src/gui/categoryfiltermodel.cpp
Normal file
|
@ -0,0 +1,442 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2016 Vladimir Golovnev <glassez@yandex.ru>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "categoryfiltermodel.h"
|
||||||
|
|
||||||
|
#include <QHash>
|
||||||
|
#include <QIcon>
|
||||||
|
|
||||||
|
#include "base/bittorrent/torrenthandle.h"
|
||||||
|
#include "base/bittorrent/session.h"
|
||||||
|
#include "guiiconprovider.h"
|
||||||
|
|
||||||
|
class CategoryModelItem
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CategoryModelItem()
|
||||||
|
: m_parent(nullptr)
|
||||||
|
, m_torrentsCount(0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryModelItem(CategoryModelItem *parent, QString categoryName, int torrentsCount = 0)
|
||||||
|
: m_parent(nullptr)
|
||||||
|
, m_name(categoryName)
|
||||||
|
, m_torrentsCount(torrentsCount)
|
||||||
|
{
|
||||||
|
if (parent)
|
||||||
|
parent->addChild(m_name, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
~CategoryModelItem()
|
||||||
|
{
|
||||||
|
clear();
|
||||||
|
if (m_parent) {
|
||||||
|
m_parent->m_torrentsCount -= m_torrentsCount;
|
||||||
|
const QString uid = m_parent->m_children.key(this);
|
||||||
|
m_parent->m_children.remove(uid);
|
||||||
|
m_parent->m_childUids.removeOne(uid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QString name() const
|
||||||
|
{
|
||||||
|
return m_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString fullName() const
|
||||||
|
{
|
||||||
|
if (!m_parent || m_parent->name().isEmpty())
|
||||||
|
return m_name;
|
||||||
|
|
||||||
|
return QString("%1/%2").arg(m_parent->fullName()).arg(m_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryModelItem *parent() const
|
||||||
|
{
|
||||||
|
return m_parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
int torrentsCount() const
|
||||||
|
{
|
||||||
|
return m_torrentsCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
void increaseTorrentsCount()
|
||||||
|
{
|
||||||
|
++m_torrentsCount;
|
||||||
|
if (m_parent)
|
||||||
|
m_parent->increaseTorrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
void decreaseTorrentsCount()
|
||||||
|
{
|
||||||
|
--m_torrentsCount;
|
||||||
|
if (m_parent)
|
||||||
|
m_parent->decreaseTorrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
int pos() const
|
||||||
|
{
|
||||||
|
if (!m_parent) return -1;
|
||||||
|
|
||||||
|
return m_parent->m_childUids.indexOf(m_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool hasChild(const QString &name) const
|
||||||
|
{
|
||||||
|
return m_children.contains(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
int childCount() const
|
||||||
|
{
|
||||||
|
return m_children.count();
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryModelItem *child(const QString &uid) const
|
||||||
|
{
|
||||||
|
return m_children.value(uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryModelItem *childAt(int index) const
|
||||||
|
{
|
||||||
|
if ((index < 0) || (index >= m_childUids.count()))
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
return m_children[m_childUids[index]];
|
||||||
|
}
|
||||||
|
|
||||||
|
void addChild(const QString &uid, CategoryModelItem *item)
|
||||||
|
{
|
||||||
|
Q_ASSERT(item);
|
||||||
|
Q_ASSERT(!item->parent());
|
||||||
|
Q_ASSERT(!m_children.contains(uid));
|
||||||
|
|
||||||
|
item->m_parent = this;
|
||||||
|
m_children[uid] = item;
|
||||||
|
auto pos = std::lower_bound(m_childUids.begin(), m_childUids.end(), uid);
|
||||||
|
m_childUids.insert(pos, uid);
|
||||||
|
m_torrentsCount += item->torrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
void clear()
|
||||||
|
{
|
||||||
|
// use copy of m_children for qDeleteAll
|
||||||
|
// to avoid collision when child removes
|
||||||
|
// itself from parent children
|
||||||
|
qDeleteAll(decltype(m_children)(m_children));
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
CategoryModelItem *m_parent;
|
||||||
|
QString m_name;
|
||||||
|
int m_torrentsCount;
|
||||||
|
QHash<QString, CategoryModelItem *> m_children;
|
||||||
|
QStringList m_childUids;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
QString shortName(const QString &fullName)
|
||||||
|
{
|
||||||
|
int pos = fullName.lastIndexOf(QLatin1Char('/'));
|
||||||
|
if (pos >= 0)
|
||||||
|
return fullName.mid(pos + 1);
|
||||||
|
return fullName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryFilterModel::CategoryFilterModel(QObject *parent)
|
||||||
|
: QAbstractItemModel(parent)
|
||||||
|
, m_rootItem(new CategoryModelItem)
|
||||||
|
{
|
||||||
|
auto session = BitTorrent::Session::instance();
|
||||||
|
|
||||||
|
connect(session, SIGNAL(categoryAdded(QString)), SLOT(categoryAdded(QString)));
|
||||||
|
connect(session, SIGNAL(categoryRemoved(QString)), SLOT(categoryRemoved(QString)));
|
||||||
|
connect(session, SIGNAL(torrentCategoryChanged(BitTorrent::TorrentHandle *const, QString))
|
||||||
|
, SLOT(torrentCategoryChanged(BitTorrent::TorrentHandle *const, QString)));
|
||||||
|
connect(session, SIGNAL(subcategoriesSupportChanged()), SLOT(subcategoriesSupportChanged()));
|
||||||
|
connect(session, SIGNAL(torrentAdded(BitTorrent::TorrentHandle *const))
|
||||||
|
, SLOT(torrentAdded(BitTorrent::TorrentHandle *const)));
|
||||||
|
connect(session, SIGNAL(torrentAboutToBeRemoved(BitTorrent::TorrentHandle *const))
|
||||||
|
, SLOT(torrentAboutToBeRemoved(BitTorrent::TorrentHandle *const)));
|
||||||
|
|
||||||
|
populate();
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryFilterModel::~CategoryFilterModel()
|
||||||
|
{
|
||||||
|
delete m_rootItem;
|
||||||
|
}
|
||||||
|
|
||||||
|
int CategoryFilterModel::columnCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant CategoryFilterModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
if (!index.isValid()) return QVariant();
|
||||||
|
|
||||||
|
auto item = static_cast<CategoryModelItem *>(index.internalPointer());
|
||||||
|
|
||||||
|
if ((index.column() == 0) && (role == Qt::DecorationRole)) {
|
||||||
|
return GuiIconProvider::instance()->getIcon("inode-directory");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((index.column() == 0) && (role == Qt::DisplayRole)) {
|
||||||
|
return QString(QStringLiteral("%1 (%2)"))
|
||||||
|
.arg(item->name()).arg(item->torrentsCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((index.column() == 0) && (role == Qt::UserRole)) {
|
||||||
|
return item->torrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
Qt::ItemFlags CategoryFilterModel::flags(const QModelIndex &index) const
|
||||||
|
{
|
||||||
|
if (!index.isValid()) return 0;
|
||||||
|
|
||||||
|
return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant CategoryFilterModel::headerData(int section, Qt::Orientation orientation, int role) const
|
||||||
|
{
|
||||||
|
if ((orientation == Qt::Horizontal) && (role == Qt::DisplayRole))
|
||||||
|
if (section == 0)
|
||||||
|
return tr("Categories");
|
||||||
|
|
||||||
|
return QVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
QModelIndex CategoryFilterModel::index(int row, int column, const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
if (column > 0)
|
||||||
|
return QModelIndex();
|
||||||
|
|
||||||
|
if (parent.isValid() && (parent.column() != 0))
|
||||||
|
return QModelIndex();
|
||||||
|
|
||||||
|
auto parentItem = parent.isValid() ? static_cast<CategoryModelItem *>(parent.internalPointer())
|
||||||
|
: m_rootItem;
|
||||||
|
if (row < parentItem->childCount())
|
||||||
|
return createIndex(row, column, parentItem->childAt(row));
|
||||||
|
|
||||||
|
return QModelIndex();
|
||||||
|
}
|
||||||
|
|
||||||
|
QModelIndex CategoryFilterModel::parent(const QModelIndex &index) const
|
||||||
|
{
|
||||||
|
if (!index.isValid())
|
||||||
|
return QModelIndex();
|
||||||
|
|
||||||
|
auto item = static_cast<CategoryModelItem *>(index.internalPointer());
|
||||||
|
if (!item) return QModelIndex();
|
||||||
|
|
||||||
|
return this->index(item->parent());
|
||||||
|
}
|
||||||
|
|
||||||
|
int CategoryFilterModel::rowCount(const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
if (parent.column() > 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (!parent.isValid())
|
||||||
|
return m_rootItem->childCount();
|
||||||
|
|
||||||
|
auto item = static_cast<CategoryModelItem *>(parent.internalPointer());
|
||||||
|
if (!item) return 0;
|
||||||
|
|
||||||
|
return item->childCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
QModelIndex CategoryFilterModel::index(const QString &categoryName) const
|
||||||
|
{
|
||||||
|
return index(findItem(categoryName));
|
||||||
|
}
|
||||||
|
|
||||||
|
QString CategoryFilterModel::categoryName(const QModelIndex &index) const
|
||||||
|
{
|
||||||
|
if (!index.isValid()) return QString();
|
||||||
|
return static_cast<CategoryModelItem *>(index.internalPointer())->fullName();
|
||||||
|
}
|
||||||
|
|
||||||
|
QModelIndex CategoryFilterModel::index(CategoryModelItem *item) const
|
||||||
|
{
|
||||||
|
if (!item || !item->parent()) return QModelIndex();
|
||||||
|
|
||||||
|
return index(item->pos(), 0, index(item->parent()));
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::categoryAdded(const QString &categoryName)
|
||||||
|
{
|
||||||
|
CategoryModelItem *parent = m_rootItem;
|
||||||
|
|
||||||
|
if (m_isSubcategoriesEnabled) {
|
||||||
|
QStringList expanded = BitTorrent::Session::expandCategory(categoryName);
|
||||||
|
if (expanded.count() > 1)
|
||||||
|
parent = findItem(expanded[expanded.count() - 2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto item = new CategoryModelItem(
|
||||||
|
parent, m_isSubcategoriesEnabled ? shortName(categoryName) : categoryName);
|
||||||
|
|
||||||
|
QModelIndex i = index(item);
|
||||||
|
beginInsertRows(i.parent(), i.row(), i.row());
|
||||||
|
endInsertRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::categoryRemoved(const QString &categoryName)
|
||||||
|
{
|
||||||
|
auto item = findItem(categoryName);
|
||||||
|
if (item) {
|
||||||
|
QModelIndex i = index(item);
|
||||||
|
beginRemoveRows(i.parent(), i.row(), i.row());
|
||||||
|
delete item;
|
||||||
|
endRemoveRows();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::torrentAdded(BitTorrent::TorrentHandle *const torrent)
|
||||||
|
{
|
||||||
|
CategoryModelItem *item = findItem(torrent->category());
|
||||||
|
Q_ASSERT(item);
|
||||||
|
|
||||||
|
item->increaseTorrentsCount();
|
||||||
|
m_rootItem->childAt(0)->increaseTorrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::torrentAboutToBeRemoved(BitTorrent::TorrentHandle *const torrent)
|
||||||
|
{
|
||||||
|
CategoryModelItem *item = findItem(torrent->category());
|
||||||
|
Q_ASSERT(item);
|
||||||
|
|
||||||
|
item->decreaseTorrentsCount();
|
||||||
|
m_rootItem->childAt(0)->decreaseTorrentsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::torrentCategoryChanged(BitTorrent::TorrentHandle *const torrent, const QString &oldCategory)
|
||||||
|
{
|
||||||
|
QModelIndex i;
|
||||||
|
|
||||||
|
auto item = findItem(oldCategory);
|
||||||
|
Q_ASSERT(item);
|
||||||
|
|
||||||
|
item->decreaseTorrentsCount();
|
||||||
|
i = index(item);
|
||||||
|
while (i.isValid()) {
|
||||||
|
emit dataChanged(i, i);
|
||||||
|
i = parent(i);
|
||||||
|
}
|
||||||
|
|
||||||
|
item = findItem(torrent->category());
|
||||||
|
Q_ASSERT(item);
|
||||||
|
|
||||||
|
item->increaseTorrentsCount();
|
||||||
|
i = index(item);
|
||||||
|
while (i.isValid()) {
|
||||||
|
emit dataChanged(i, i);
|
||||||
|
i = parent(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::subcategoriesSupportChanged()
|
||||||
|
{
|
||||||
|
beginResetModel();
|
||||||
|
populate();
|
||||||
|
endResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterModel::populate()
|
||||||
|
{
|
||||||
|
m_rootItem->clear();
|
||||||
|
|
||||||
|
auto session = BitTorrent::Session::instance();
|
||||||
|
auto torrents = session->torrents();
|
||||||
|
m_isSubcategoriesEnabled = session->isSubcategoriesEnabled();
|
||||||
|
|
||||||
|
const QString UID_ALL;
|
||||||
|
const QString UID_UNCATEGORIZED(QChar(1));
|
||||||
|
|
||||||
|
// All torrents
|
||||||
|
m_rootItem->addChild(UID_ALL, new CategoryModelItem(nullptr, tr("All"), torrents.count()));
|
||||||
|
|
||||||
|
// Uncategorized torrents
|
||||||
|
using Torrent = BitTorrent::TorrentHandle;
|
||||||
|
m_rootItem->addChild(
|
||||||
|
UID_UNCATEGORIZED
|
||||||
|
, new CategoryModelItem(
|
||||||
|
nullptr, tr("Uncategorized")
|
||||||
|
, std::count_if(torrents.begin(), torrents.end()
|
||||||
|
, [](Torrent *torrent) { return torrent->category().isEmpty(); })));
|
||||||
|
|
||||||
|
using Torrent = BitTorrent::TorrentHandle;
|
||||||
|
foreach (const QString &category, session->categories()) {
|
||||||
|
if (m_isSubcategoriesEnabled) {
|
||||||
|
CategoryModelItem *parent = m_rootItem;
|
||||||
|
foreach (const QString &subcat, session->expandCategory(category)) {
|
||||||
|
const QString subcatName = shortName(subcat);
|
||||||
|
if (!parent->hasChild(subcatName)) {
|
||||||
|
new CategoryModelItem(
|
||||||
|
parent, subcatName
|
||||||
|
, std::count_if(torrents.begin(), torrents.end()
|
||||||
|
, [subcat](Torrent *torrent) { return torrent->category() == subcat; }));
|
||||||
|
}
|
||||||
|
parent = parent->child(subcatName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
new CategoryModelItem(
|
||||||
|
m_rootItem, category
|
||||||
|
, std::count_if(torrents.begin(), torrents.end()
|
||||||
|
, [category](Torrent *torrent) { return torrent->belongsToCategory(category); }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryModelItem *CategoryFilterModel::findItem(const QString &fullName) const
|
||||||
|
{
|
||||||
|
if (fullName.isEmpty())
|
||||||
|
return m_rootItem->childAt(1); // "Uncategorized" item
|
||||||
|
|
||||||
|
if (!m_isSubcategoriesEnabled)
|
||||||
|
return m_rootItem->child(fullName);
|
||||||
|
|
||||||
|
CategoryModelItem *item = m_rootItem;
|
||||||
|
foreach (const QString &subcat, BitTorrent::Session::expandCategory(fullName)) {
|
||||||
|
const QString subcatName = shortName(subcat);
|
||||||
|
if (!item->hasChild(subcatName)) return nullptr;
|
||||||
|
item = item->child(subcatName);
|
||||||
|
}
|
||||||
|
|
||||||
|
return item;
|
||||||
|
}
|
79
src/gui/categoryfiltermodel.h
Normal file
79
src/gui/categoryfiltermodel.h
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2016 Vladimir Golovnev <glassez@yandex.ru>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef CATEGORYFILTERMODEL_H
|
||||||
|
#define CATEGORYFILTERMODEL_H
|
||||||
|
|
||||||
|
#include <QAbstractItemModel>
|
||||||
|
#include <QHash>
|
||||||
|
#include <QModelIndex>
|
||||||
|
|
||||||
|
namespace BitTorrent
|
||||||
|
{
|
||||||
|
class TorrentHandle;
|
||||||
|
}
|
||||||
|
|
||||||
|
class CategoryModelItem;
|
||||||
|
|
||||||
|
class CategoryFilterModel: public QAbstractItemModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit CategoryFilterModel(QObject *parent = nullptr);
|
||||||
|
~CategoryFilterModel();
|
||||||
|
|
||||||
|
int columnCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
|
Qt::ItemFlags flags(const QModelIndex &index) const override;
|
||||||
|
QVariant headerData(int section, Qt::Orientation orientation, int role) const override;
|
||||||
|
QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
QModelIndex parent(const QModelIndex &index) const override;
|
||||||
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
|
||||||
|
QModelIndex index(const QString &categoryName) const;
|
||||||
|
QString categoryName(const QModelIndex &index) const;
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void categoryAdded(const QString &categoryName);
|
||||||
|
void categoryRemoved(const QString &categoryName);
|
||||||
|
void torrentAdded(BitTorrent::TorrentHandle *const torrent);
|
||||||
|
void torrentAboutToBeRemoved(BitTorrent::TorrentHandle *const torrent);
|
||||||
|
void torrentCategoryChanged(BitTorrent::TorrentHandle *const torrent, const QString &oldCategory);
|
||||||
|
void subcategoriesSupportChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void populate();
|
||||||
|
QModelIndex index(CategoryModelItem *item) const;
|
||||||
|
CategoryModelItem *findItem(const QString &fullName) const;
|
||||||
|
|
||||||
|
bool m_isSubcategoriesEnabled;
|
||||||
|
CategoryModelItem *m_rootItem;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // CATEGORYFILTERMODEL_H
|
269
src/gui/categoryfilterwidget.cpp
Normal file
269
src/gui/categoryfilterwidget.cpp
Normal file
|
@ -0,0 +1,269 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2016 Vladimir Golovnev <glassez@yandex.ru>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "categoryfilterwidget.h"
|
||||||
|
|
||||||
|
#include <QAction>
|
||||||
|
#include <QHeaderView>
|
||||||
|
#include <QLayout>
|
||||||
|
#include <QMenu>
|
||||||
|
#include <QMessageBox>
|
||||||
|
|
||||||
|
#include "base/bittorrent/session.h"
|
||||||
|
#include "base/utils/misc.h"
|
||||||
|
#include "autoexpandabledialog.h"
|
||||||
|
#include "categoryfiltermodel.h"
|
||||||
|
#include "guiiconprovider.h"
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
QString getCategoryFilter(const CategoryFilterModel *const model, const QModelIndex &index)
|
||||||
|
{
|
||||||
|
QString categoryFilter; // Defaults to All
|
||||||
|
if (index.isValid()) {
|
||||||
|
if (!index.parent().isValid() && (index.row() == 1))
|
||||||
|
categoryFilter = ""; // Uncategorized
|
||||||
|
else if (index.parent().isValid() || (index.row() > 1))
|
||||||
|
categoryFilter = model->categoryName(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
return categoryFilter;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isSpecialItem(const QModelIndex &index)
|
||||||
|
{
|
||||||
|
// the first two items at first level are special items:
|
||||||
|
// 'All' and 'Uncategorized'
|
||||||
|
return (!index.parent().isValid() && (index.row() <= 1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CategoryFilterWidget::CategoryFilterWidget(QWidget *parent)
|
||||||
|
: QTreeView(parent)
|
||||||
|
{
|
||||||
|
setModel(new CategoryFilterModel(this));
|
||||||
|
setFrameShape(QFrame::NoFrame);
|
||||||
|
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
||||||
|
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
||||||
|
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
|
||||||
|
setUniformRowHeights(true);
|
||||||
|
setHeaderHidden(true);
|
||||||
|
setIconSize(Utils::Misc::smallIconSize());
|
||||||
|
#if defined(Q_OS_MAC)
|
||||||
|
setAttribute(Qt::WA_MacShowFocusRect, false);
|
||||||
|
#endif
|
||||||
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
||||||
|
|
||||||
|
connect(this, SIGNAL(collapsed(QModelIndex)), SLOT(callUpdateGeometry()));
|
||||||
|
connect(this, SIGNAL(expanded(QModelIndex)), SLOT(callUpdateGeometry()));
|
||||||
|
connect(this, SIGNAL(customContextMenuRequested(QPoint)), SLOT(showMenu(QPoint)));
|
||||||
|
connect(selectionModel(), SIGNAL(currentRowChanged(QModelIndex,QModelIndex))
|
||||||
|
, SLOT(onCurrentRowChanged(QModelIndex,QModelIndex)));
|
||||||
|
connect(model(), SIGNAL(modelReset()), SLOT(callUpdateGeometry()));
|
||||||
|
}
|
||||||
|
|
||||||
|
QString CategoryFilterWidget::currentCategory() const
|
||||||
|
{
|
||||||
|
QModelIndex current;
|
||||||
|
auto selectedRows = selectionModel()->selectedRows();
|
||||||
|
if (!selectedRows.isEmpty())
|
||||||
|
current = selectedRows.first();
|
||||||
|
|
||||||
|
return getCategoryFilter(static_cast<CategoryFilterModel *>(model()), current);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::onCurrentRowChanged(const QModelIndex ¤t, const QModelIndex &previous)
|
||||||
|
{
|
||||||
|
Q_UNUSED(previous);
|
||||||
|
|
||||||
|
emit categoryChanged(getCategoryFilter(static_cast<CategoryFilterModel *>(model()), current));
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::showMenu(QPoint)
|
||||||
|
{
|
||||||
|
QMenu menu(this);
|
||||||
|
|
||||||
|
QAction *addAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("list-add")
|
||||||
|
, tr("Add category..."));
|
||||||
|
connect(addAct, SIGNAL(triggered()), SLOT(addCategory()));
|
||||||
|
|
||||||
|
auto selectedRows = selectionModel()->selectedRows();
|
||||||
|
if (!selectedRows.empty() && !isSpecialItem(selectedRows.first())) {
|
||||||
|
if (BitTorrent::Session::instance()->isSubcategoriesEnabled()) {
|
||||||
|
QAction *addSubAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("list-add")
|
||||||
|
, tr("Add subcategory..."));
|
||||||
|
connect(addSubAct, SIGNAL(triggered()), SLOT(addSubcategory()));
|
||||||
|
}
|
||||||
|
|
||||||
|
QAction *removeAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("list-remove")
|
||||||
|
, tr("Remove category"));
|
||||||
|
connect(removeAct, SIGNAL(triggered()), SLOT(removeCategory()));
|
||||||
|
}
|
||||||
|
|
||||||
|
QAction *removeUnusedAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("list-remove")
|
||||||
|
, tr("Remove unused categories"));
|
||||||
|
connect(removeUnusedAct, SIGNAL(triggered()), SLOT(removeUnusedCategories()));
|
||||||
|
|
||||||
|
menu.addSeparator();
|
||||||
|
|
||||||
|
QAction *startAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("media-playback-start")
|
||||||
|
, tr("Resume torrents"));
|
||||||
|
connect(startAct, SIGNAL(triggered()), SIGNAL(actionResumeTorrentsTriggered()));
|
||||||
|
|
||||||
|
QAction *pauseAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("media-playback-pause")
|
||||||
|
, tr("Pause torrents"));
|
||||||
|
connect(pauseAct, SIGNAL(triggered()), SIGNAL(actionPauseTorrentsTriggered()));
|
||||||
|
|
||||||
|
QAction *deleteTorrentsAct = menu.addAction(
|
||||||
|
GuiIconProvider::instance()->getIcon("edit-delete")
|
||||||
|
, tr("Delete torrents"));
|
||||||
|
connect(deleteTorrentsAct, SIGNAL(triggered()), SIGNAL(actionDeleteTorrentsTriggered()));
|
||||||
|
|
||||||
|
menu.exec(QCursor::pos());
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::callUpdateGeometry()
|
||||||
|
{
|
||||||
|
updateGeometry();
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize CategoryFilterWidget::sizeHint() const
|
||||||
|
{
|
||||||
|
#ifdef QBT_USES_QT5
|
||||||
|
return viewportSizeHint();
|
||||||
|
#else
|
||||||
|
int lastRow = model()->rowCount() - 1;
|
||||||
|
QModelIndex last = model()->index(lastRow, 0);
|
||||||
|
while ((lastRow >= 0) && isExpanded(last)) {
|
||||||
|
lastRow = model()->rowCount(last) - 1;
|
||||||
|
last = model()->index(lastRow, 0, last);
|
||||||
|
}
|
||||||
|
const QRect deepestRect = visualRect(last);
|
||||||
|
|
||||||
|
if (!deepestRect.isValid())
|
||||||
|
return viewport()->sizeHint();
|
||||||
|
|
||||||
|
return QSize(header()->length(), deepestRect.bottom() + 1);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize CategoryFilterWidget::minimumSizeHint() const
|
||||||
|
{
|
||||||
|
QSize size = sizeHint();
|
||||||
|
size.setWidth(6);
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::rowsInserted(const QModelIndex &parent, int start, int end)
|
||||||
|
{
|
||||||
|
QTreeView::rowsInserted(parent, start, end);
|
||||||
|
|
||||||
|
// Expand all parents if the parent(s) of the node are not expanded.
|
||||||
|
QModelIndex p = parent;
|
||||||
|
while (p.isValid()) {
|
||||||
|
if (!isExpanded(p))
|
||||||
|
expand(p);
|
||||||
|
p = model()->parent(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
updateGeometry();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString CategoryFilterWidget::askCategoryName()
|
||||||
|
{
|
||||||
|
bool ok;
|
||||||
|
QString category = "";
|
||||||
|
bool invalid;
|
||||||
|
do {
|
||||||
|
invalid = false;
|
||||||
|
category = AutoExpandableDialog::getText(
|
||||||
|
this, tr("New Category"), tr("Category:"), QLineEdit::Normal, category, &ok);
|
||||||
|
if (ok && !category.isEmpty()) {
|
||||||
|
if (!BitTorrent::Session::isValidCategoryName(category)) {
|
||||||
|
QMessageBox::warning(
|
||||||
|
this, tr("Invalid category name")
|
||||||
|
, tr("Category name must not contain '\\'.\n"
|
||||||
|
"Category name must not start/end with '/'.\n"
|
||||||
|
"Category name must not contain '//' sequence."));
|
||||||
|
invalid = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while (invalid);
|
||||||
|
|
||||||
|
return ok ? category : QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::addCategory()
|
||||||
|
{
|
||||||
|
const QString category = askCategoryName();
|
||||||
|
if (category.isEmpty()) return;
|
||||||
|
|
||||||
|
if (BitTorrent::Session::instance()->categories().contains(category))
|
||||||
|
QMessageBox::warning(this, tr("Category exists"), tr("Category name already exists."));
|
||||||
|
else
|
||||||
|
BitTorrent::Session::instance()->addCategory(category);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::addSubcategory()
|
||||||
|
{
|
||||||
|
const QString subcat = askCategoryName();
|
||||||
|
if (subcat.isEmpty()) return;
|
||||||
|
|
||||||
|
const QString category = QString(QStringLiteral("%1/%2")).arg(currentCategory()).arg(subcat);
|
||||||
|
|
||||||
|
if (BitTorrent::Session::instance()->categories().contains(category))
|
||||||
|
QMessageBox::warning(this, tr("Category exists")
|
||||||
|
, tr("Subcategory name already exists in selected category."));
|
||||||
|
else
|
||||||
|
BitTorrent::Session::instance()->addCategory(category);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::removeCategory()
|
||||||
|
{
|
||||||
|
auto selectedRows = selectionModel()->selectedRows();
|
||||||
|
if (!selectedRows.empty() && !isSpecialItem(selectedRows.first())) {
|
||||||
|
BitTorrent::Session::instance()->removeCategory(
|
||||||
|
static_cast<CategoryFilterModel *>(model())->categoryName(selectedRows.first()));
|
||||||
|
updateGeometry();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CategoryFilterWidget::removeUnusedCategories()
|
||||||
|
{
|
||||||
|
auto session = BitTorrent::Session::instance();
|
||||||
|
foreach (const QString &category, session->categories())
|
||||||
|
if (model()->data(static_cast<CategoryFilterModel *>(model())->index(category), Qt::UserRole) == 0)
|
||||||
|
session->removeCategory(category);
|
||||||
|
updateGeometry();
|
||||||
|
}
|
60
src/gui/categoryfilterwidget.h
Normal file
60
src/gui/categoryfilterwidget.h
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2016 Vladimir Golovnev <glassez@yandex.ru>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <QTreeView>
|
||||||
|
|
||||||
|
class CategoryFilterWidget: public QTreeView
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit CategoryFilterWidget(QWidget *parent = nullptr);
|
||||||
|
|
||||||
|
QString currentCategory() const;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void categoryChanged(const QString &categoryName);
|
||||||
|
void actionResumeTorrentsTriggered();
|
||||||
|
void actionPauseTorrentsTriggered();
|
||||||
|
void actionDeleteTorrentsTriggered();
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void onCurrentRowChanged(const QModelIndex ¤t, const QModelIndex &previous);
|
||||||
|
void showMenu(QPoint);
|
||||||
|
void callUpdateGeometry();
|
||||||
|
void addCategory();
|
||||||
|
void addSubcategory();
|
||||||
|
void removeCategory();
|
||||||
|
void removeUnusedCategories();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QSize sizeHint() const override;
|
||||||
|
QSize minimumSizeHint() const override;
|
||||||
|
void rowsInserted(const QModelIndex &parent, int start, int end) override;
|
||||||
|
QString askCategoryName();
|
||||||
|
};
|
|
@ -49,7 +49,9 @@ HEADERS += \
|
||||||
$$PWD/search/searchlistdelegate.h \
|
$$PWD/search/searchlistdelegate.h \
|
||||||
$$PWD/search/searchsortmodel.h \
|
$$PWD/search/searchsortmodel.h \
|
||||||
$$PWD/cookiesmodel.h \
|
$$PWD/cookiesmodel.h \
|
||||||
$$PWD/cookiesdialog.h
|
$$PWD/cookiesdialog.h \
|
||||||
|
$$PWD/categoryfiltermodel.h \
|
||||||
|
$$PWD/categoryfilterwidget.h
|
||||||
|
|
||||||
SOURCES += \
|
SOURCES += \
|
||||||
$$PWD/mainwindow.cpp \
|
$$PWD/mainwindow.cpp \
|
||||||
|
@ -89,7 +91,9 @@ SOURCES += \
|
||||||
$$PWD/search/searchlistdelegate.cpp \
|
$$PWD/search/searchlistdelegate.cpp \
|
||||||
$$PWD/search/searchsortmodel.cpp \
|
$$PWD/search/searchsortmodel.cpp \
|
||||||
$$PWD/cookiesmodel.cpp \
|
$$PWD/cookiesmodel.cpp \
|
||||||
$$PWD/cookiesdialog.cpp
|
$$PWD/cookiesdialog.cpp \
|
||||||
|
$$PWD/categoryfiltermodel.cpp \
|
||||||
|
$$PWD/categoryfilterwidget.cpp
|
||||||
|
|
||||||
win32|macx {
|
win32|macx {
|
||||||
HEADERS += $$PWD/programupdater.h
|
HEADERS += $$PWD/programupdater.h
|
||||||
|
|
|
@ -30,31 +30,32 @@
|
||||||
|
|
||||||
#include "transferlistfilterswidget.h"
|
#include "transferlistfilterswidget.h"
|
||||||
|
|
||||||
|
#include <QCheckBox>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QListWidgetItem>
|
|
||||||
#include <QIcon>
|
#include <QIcon>
|
||||||
#include <QVBoxLayout>
|
#include <QListWidgetItem>
|
||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QCheckBox>
|
|
||||||
#include <QScrollArea>
|
#include <QScrollArea>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
#include "transferlistdelegate.h"
|
|
||||||
#include "transferlistwidget.h"
|
|
||||||
#include "base/preferences.h"
|
|
||||||
#include "torrentmodel.h"
|
|
||||||
#include "guiiconprovider.h"
|
|
||||||
#include "base/utils/fs.h"
|
|
||||||
#include "base/utils/string.h"
|
|
||||||
#include "autoexpandabledialog.h"
|
|
||||||
#include "base/torrentfilter.h"
|
|
||||||
#include "base/bittorrent/trackerentry.h"
|
|
||||||
#include "base/bittorrent/session.h"
|
#include "base/bittorrent/session.h"
|
||||||
#include "base/bittorrent/torrenthandle.h"
|
#include "base/bittorrent/torrenthandle.h"
|
||||||
|
#include "base/bittorrent/trackerentry.h"
|
||||||
|
#include "base/logger.h"
|
||||||
#include "base/net/downloadmanager.h"
|
#include "base/net/downloadmanager.h"
|
||||||
#include "base/net/downloadhandler.h"
|
#include "base/net/downloadhandler.h"
|
||||||
|
#include "base/preferences.h"
|
||||||
|
#include "base/torrentfilter.h"
|
||||||
|
#include "base/utils/fs.h"
|
||||||
#include "base/utils/misc.h"
|
#include "base/utils/misc.h"
|
||||||
#include "base/logger.h"
|
#include "base/utils/string.h"
|
||||||
|
#include "autoexpandabledialog.h"
|
||||||
|
#include "categoryfilterwidget.h"
|
||||||
|
#include "guiiconprovider.h"
|
||||||
|
#include "torrentmodel.h"
|
||||||
|
#include "transferlistdelegate.h"
|
||||||
|
#include "transferlistwidget.h"
|
||||||
|
|
||||||
FiltersBase::FiltersBase(QWidget *parent, TransferListWidget *transferList)
|
FiltersBase::FiltersBase(QWidget *parent, TransferListWidget *transferList)
|
||||||
: QListWidget(parent)
|
: QListWidget(parent)
|
||||||
|
@ -177,266 +178,6 @@ void StatusFiltersWidget::handleNewTorrent(BitTorrent::TorrentHandle *const) {}
|
||||||
|
|
||||||
void StatusFiltersWidget::torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const) {}
|
void StatusFiltersWidget::torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const) {}
|
||||||
|
|
||||||
CategoryFiltersList::CategoryFiltersList(QWidget *parent, TransferListWidget *transferList)
|
|
||||||
: FiltersBase(parent, transferList)
|
|
||||||
{
|
|
||||||
connect(BitTorrent::Session::instance(), SIGNAL(torrentCategoryChanged(BitTorrent::TorrentHandle *const, QString)), SLOT(torrentCategoryChanged(BitTorrent::TorrentHandle *const, QString)));
|
|
||||||
connect(BitTorrent::Session::instance(), SIGNAL(categoryAdded(QString)), SLOT(addItem(QString)));
|
|
||||||
connect(BitTorrent::Session::instance(), SIGNAL(categoryRemoved(QString)), SLOT(categoryRemoved(QString)));
|
|
||||||
connect(BitTorrent::Session::instance(), SIGNAL(subcategoriesSupportChanged()), SLOT(subcategoriesSupportChanged()));
|
|
||||||
|
|
||||||
refresh();
|
|
||||||
toggleFilter(Preferences::instance()->getCategoryFilterState());
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::refresh()
|
|
||||||
{
|
|
||||||
clear();
|
|
||||||
m_categories.clear();
|
|
||||||
m_totalTorrents = 0;
|
|
||||||
m_totalCategorized = 0;
|
|
||||||
|
|
||||||
QListWidgetItem *allCategories = new QListWidgetItem(this);
|
|
||||||
allCategories->setData(Qt::DisplayRole, QVariant(tr("All (0)", "this is for the category filter")));
|
|
||||||
allCategories->setData(Qt::DecorationRole, GuiIconProvider::instance()->getIcon("inode-directory"));
|
|
||||||
QListWidgetItem *noCategory = new QListWidgetItem(this);
|
|
||||||
noCategory->setData(Qt::DisplayRole, QVariant(tr("Uncategorized (0)")));
|
|
||||||
noCategory->setData(Qt::DecorationRole, GuiIconProvider::instance()->getIcon("inode-directory"));
|
|
||||||
|
|
||||||
foreach (const QString &category, BitTorrent::Session::instance()->categories())
|
|
||||||
addItem(category, false);
|
|
||||||
|
|
||||||
foreach (BitTorrent::TorrentHandle *const torrent, BitTorrent::Session::instance()->torrents())
|
|
||||||
handleNewTorrent(torrent);
|
|
||||||
|
|
||||||
setCurrentRow(0, QItemSelectionModel::SelectCurrent);
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::addItem(const QString &category, bool hasTorrent)
|
|
||||||
{
|
|
||||||
if (category.isEmpty()) return;
|
|
||||||
|
|
||||||
int torrentsInCategory = 0;
|
|
||||||
QListWidgetItem *categoryItem = 0;
|
|
||||||
|
|
||||||
bool exists = m_categories.contains(category);
|
|
||||||
if (exists) {
|
|
||||||
torrentsInCategory = m_categories.value(category);
|
|
||||||
categoryItem = item(rowFromCategory(category));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
categoryItem = new QListWidgetItem();
|
|
||||||
categoryItem->setData(Qt::DecorationRole, GuiIconProvider::instance()->getIcon("inode-directory"));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hasTorrent)
|
|
||||||
++torrentsInCategory;
|
|
||||||
|
|
||||||
m_categories.insert(category, torrentsInCategory);
|
|
||||||
categoryItem->setText(QString("%1 (%2)").arg(category).arg(torrentsInCategory));
|
|
||||||
if (exists) return;
|
|
||||||
|
|
||||||
Q_ASSERT(count() >= 2);
|
|
||||||
int insPos = count();
|
|
||||||
for (int i = 2; i < count(); ++i) {
|
|
||||||
if (Utils::String::naturalCompareCaseSensitive(category, item(i)->text())) {
|
|
||||||
insPos = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
QListWidget::insertItem(insPos, categoryItem);
|
|
||||||
updateGeometry();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::removeItem(const QString &category)
|
|
||||||
{
|
|
||||||
if (category.isEmpty()) return;
|
|
||||||
|
|
||||||
int torrentsInCategory = m_categories.value(category) - 1;
|
|
||||||
int row = rowFromCategory(category);
|
|
||||||
if (row < 2) return;
|
|
||||||
|
|
||||||
QListWidgetItem *categoryItem = item(row);
|
|
||||||
categoryItem->setText(QString("%1 (%2)").arg(category).arg(torrentsInCategory));
|
|
||||||
m_categories.insert(category, torrentsInCategory);
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::removeSelectedCategory()
|
|
||||||
{
|
|
||||||
QList<QListWidgetItem*> items = selectedItems();
|
|
||||||
if (items.size() == 0) return;
|
|
||||||
|
|
||||||
const int categoryRow = row(items.first());
|
|
||||||
if (categoryRow < 2) return;
|
|
||||||
|
|
||||||
BitTorrent::Session::instance()->removeCategory(categoryFromRow(categoryRow));
|
|
||||||
updateGeometry();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::removeUnusedCategories()
|
|
||||||
{
|
|
||||||
foreach (const QString &category, m_categories.keys())
|
|
||||||
if (m_categories[category] == 0)
|
|
||||||
BitTorrent::Session::instance()->removeCategory(category);
|
|
||||||
updateGeometry();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::torrentCategoryChanged(BitTorrent::TorrentHandle *const torrent, const QString &oldCategory)
|
|
||||||
{
|
|
||||||
qDebug() << "Torrent category changed from" << oldCategory << "to" << torrent->category();
|
|
||||||
|
|
||||||
if (torrent->category().isEmpty() && !oldCategory.isEmpty())
|
|
||||||
--m_totalCategorized;
|
|
||||||
else if (!torrent->category().isEmpty() && oldCategory.isEmpty())
|
|
||||||
++m_totalCategorized;
|
|
||||||
|
|
||||||
item(1)->setText(tr("Uncategorized (%1)").arg(m_totalTorrents - m_totalCategorized));
|
|
||||||
|
|
||||||
if (BitTorrent::Session::instance()->isSubcategoriesEnabled()) {
|
|
||||||
foreach (const QString &subcategory, BitTorrent::Session::expandCategory(oldCategory))
|
|
||||||
removeItem(subcategory);
|
|
||||||
foreach (const QString &subcategory, BitTorrent::Session::expandCategory(torrent->category()))
|
|
||||||
addItem(subcategory, true);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
removeItem(oldCategory);
|
|
||||||
addItem(torrent->category(), true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::categoryRemoved(const QString &category)
|
|
||||||
{
|
|
||||||
m_categories.remove(category);
|
|
||||||
delete takeItem(rowFromCategory(category));
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::subcategoriesSupportChanged()
|
|
||||||
{
|
|
||||||
refresh();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::showMenu(QPoint)
|
|
||||||
{
|
|
||||||
QMenu menu(this);
|
|
||||||
QAction *addAct = menu.addAction(GuiIconProvider::instance()->getIcon("list-add"), tr("Add category..."));
|
|
||||||
QAction *removeAct = 0;
|
|
||||||
QAction *removeUnusedAct = 0;
|
|
||||||
if (!selectedItems().empty() && row(selectedItems().first()) > 1)
|
|
||||||
removeAct = menu.addAction(GuiIconProvider::instance()->getIcon("list-remove"), tr("Remove category"));
|
|
||||||
removeUnusedAct = menu.addAction(GuiIconProvider::instance()->getIcon("list-remove"), tr("Remove unused categories"));
|
|
||||||
menu.addSeparator();
|
|
||||||
QAction *startAct = menu.addAction(GuiIconProvider::instance()->getIcon("media-playback-start"), tr("Resume torrents"));
|
|
||||||
QAction *pauseAct = menu.addAction(GuiIconProvider::instance()->getIcon("media-playback-pause"), tr("Pause torrents"));
|
|
||||||
QAction *deleteTorrentsAct = menu.addAction(GuiIconProvider::instance()->getIcon("edit-delete"), tr("Delete torrents"));
|
|
||||||
QAction *act = 0;
|
|
||||||
act = menu.exec(QCursor::pos());
|
|
||||||
if (!act)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (act == removeAct) {
|
|
||||||
removeSelectedCategory();
|
|
||||||
}
|
|
||||||
else if (act == removeUnusedAct) {
|
|
||||||
removeUnusedCategories();
|
|
||||||
}
|
|
||||||
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 category = "";
|
|
||||||
bool invalid;
|
|
||||||
do {
|
|
||||||
invalid = false;
|
|
||||||
category = AutoExpandableDialog::getText(this, tr("New Category"), tr("Category:"), QLineEdit::Normal, category, &ok);
|
|
||||||
if (ok && !category.isEmpty()) {
|
|
||||||
if (!BitTorrent::Session::isValidCategoryName(category)) {
|
|
||||||
QMessageBox::warning(this, tr("Invalid category name"),
|
|
||||||
tr("Category name must not contain '\\'.\n"
|
|
||||||
"Category name must not start/end with '/'.\n"
|
|
||||||
"Category name must not contain '//' sequence."));
|
|
||||||
invalid = true;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
BitTorrent::Session::instance()->addCategory(category);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while (invalid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::applyFilter(int row)
|
|
||||||
{
|
|
||||||
if (row >= 0)
|
|
||||||
transferList->applyCategoryFilter(categoryFromRow(row));
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::handleNewTorrent(BitTorrent::TorrentHandle *const torrent)
|
|
||||||
{
|
|
||||||
Q_ASSERT(torrent);
|
|
||||||
|
|
||||||
++m_totalTorrents;
|
|
||||||
if (!torrent->category().isEmpty())
|
|
||||||
++m_totalCategorized;
|
|
||||||
|
|
||||||
item(0)->setText(tr("All (%1)", "this is for the category filter").arg(m_totalTorrents));
|
|
||||||
item(1)->setText(tr("Uncategorized (%1)").arg(m_totalTorrents - m_totalCategorized));
|
|
||||||
|
|
||||||
if (BitTorrent::Session::instance()->isSubcategoriesEnabled()) {
|
|
||||||
foreach (const QString &subcategory, BitTorrent::Session::expandCategory(torrent->category()))
|
|
||||||
addItem(subcategory, true);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
addItem(torrent->category(), true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void CategoryFiltersList::torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const torrent)
|
|
||||||
{
|
|
||||||
Q_ASSERT(torrent);
|
|
||||||
|
|
||||||
--m_totalTorrents;
|
|
||||||
if (!torrent->category().isEmpty())
|
|
||||||
--m_totalCategorized;
|
|
||||||
|
|
||||||
item(0)->setText(tr("All (%1)", "this is for the category filter").arg(m_totalTorrents));
|
|
||||||
item(1)->setText(tr("Uncategorized (%1)").arg(m_totalTorrents - m_totalCategorized));
|
|
||||||
|
|
||||||
if (BitTorrent::Session::instance()->isSubcategoriesEnabled()) {
|
|
||||||
foreach (const QString &subcategory, BitTorrent::Session::expandCategory(torrent->category()))
|
|
||||||
removeItem(subcategory);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
removeItem(torrent->category());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
QString CategoryFiltersList::categoryFromRow(int row) const
|
|
||||||
{
|
|
||||||
if (row == 0) return QString(); // All
|
|
||||||
if (row == 1) return QLatin1String(""); // Uncategorized
|
|
||||||
|
|
||||||
const QString &category = item(row)->text();
|
|
||||||
QStringList parts = category.split(" ");
|
|
||||||
Q_ASSERT(parts.size() >= 2);
|
|
||||||
parts.removeLast(); // Remove trailing number
|
|
||||||
return parts.join(" ");
|
|
||||||
}
|
|
||||||
|
|
||||||
int CategoryFiltersList::rowFromCategory(const QString &category) const
|
|
||||||
{
|
|
||||||
Q_ASSERT(!category.isEmpty());
|
|
||||||
for (int i = 2; i<count(); ++i)
|
|
||||||
if (category == categoryFromRow(i)) return i;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
TrackerFiltersList::TrackerFiltersList(QWidget *parent, TransferListWidget *transferList)
|
TrackerFiltersList::TrackerFiltersList(QWidget *parent, TransferListWidget *transferList)
|
||||||
: FiltersBase(parent, transferList)
|
: FiltersBase(parent, transferList)
|
||||||
, m_totalTorrents(0)
|
, m_totalTorrents(0)
|
||||||
|
@ -546,7 +287,7 @@ void TrackerFiltersList::removeItem(const QString &tracker, const QString &hash)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (trackerItem != nullptr)
|
if (trackerItem != nullptr)
|
||||||
trackerItem->setText(tr("%1 (%2)", "openbittorrent.com (10)").arg(host).arg(tmp.size()));
|
trackerItem->setText(QString("%1 (%2)").arg(host).arg(tmp.size()));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
row = 1;
|
row = 1;
|
||||||
|
@ -787,7 +528,8 @@ QStringList TrackerFiltersList::getHashes(int row)
|
||||||
|
|
||||||
TransferListFiltersWidget::TransferListFiltersWidget(QWidget *parent, TransferListWidget *transferList)
|
TransferListFiltersWidget::TransferListFiltersWidget(QWidget *parent, TransferListWidget *transferList)
|
||||||
: QFrame(parent)
|
: QFrame(parent)
|
||||||
, trackerFilters(0)
|
, m_transferList(transferList)
|
||||||
|
, m_trackerFilters(0)
|
||||||
{
|
{
|
||||||
Preferences* const pref = Preferences::instance();
|
Preferences* const pref = Preferences::instance();
|
||||||
|
|
||||||
|
@ -826,50 +568,58 @@ TransferListFiltersWidget::TransferListFiltersWidget(QWidget *parent, TransferLi
|
||||||
QCheckBox *categoryLabel = new QCheckBox(tr("Categories"), this);
|
QCheckBox *categoryLabel = new QCheckBox(tr("Categories"), this);
|
||||||
categoryLabel->setChecked(pref->getCategoryFilterState());
|
categoryLabel->setChecked(pref->getCategoryFilterState());
|
||||||
categoryLabel->setFont(font);
|
categoryLabel->setFont(font);
|
||||||
|
connect(categoryLabel, SIGNAL(toggled(bool)), SLOT(onCategoryFilterStateChanged(bool)));
|
||||||
frameLayout->addWidget(categoryLabel);
|
frameLayout->addWidget(categoryLabel);
|
||||||
|
|
||||||
CategoryFiltersList *categoryFilters = new CategoryFiltersList(this, transferList);
|
m_categoryFilterWidget = new CategoryFilterWidget(this);
|
||||||
frameLayout->addWidget(categoryFilters);
|
connect(m_categoryFilterWidget, SIGNAL(actionDeleteTorrentsTriggered())
|
||||||
|
, transferList, SLOT(deleteVisibleTorrents()));
|
||||||
|
connect(m_categoryFilterWidget, SIGNAL(actionPauseTorrentsTriggered())
|
||||||
|
, transferList, SLOT(pauseVisibleTorrents()));
|
||||||
|
connect(m_categoryFilterWidget, SIGNAL(actionResumeTorrentsTriggered())
|
||||||
|
, transferList, SLOT(startVisibleTorrents()));
|
||||||
|
connect(m_categoryFilterWidget, SIGNAL(categoryChanged(QString))
|
||||||
|
, transferList, SLOT(applyCategoryFilter(QString)));
|
||||||
|
onCategoryFilterStateChanged(pref->getCategoryFilterState());
|
||||||
|
frameLayout->addWidget(m_categoryFilterWidget);
|
||||||
|
|
||||||
QCheckBox *trackerLabel = new QCheckBox(tr("Trackers"), this);
|
QCheckBox *trackerLabel = new QCheckBox(tr("Trackers"), this);
|
||||||
trackerLabel->setChecked(pref->getTrackerFilterState());
|
trackerLabel->setChecked(pref->getTrackerFilterState());
|
||||||
trackerLabel->setFont(font);
|
trackerLabel->setFont(font);
|
||||||
frameLayout->addWidget(trackerLabel);
|
frameLayout->addWidget(trackerLabel);
|
||||||
|
|
||||||
trackerFilters = new TrackerFiltersList(this, transferList);
|
m_trackerFilters = new TrackerFiltersList(this, transferList);
|
||||||
frameLayout->addWidget(trackerFilters);
|
frameLayout->addWidget(m_trackerFilters);
|
||||||
|
|
||||||
connect(statusLabel, SIGNAL(toggled(bool)), statusFilters, SLOT(toggleFilter(bool)));
|
connect(statusLabel, SIGNAL(toggled(bool)), statusFilters, SLOT(toggleFilter(bool)));
|
||||||
connect(statusLabel, SIGNAL(toggled(bool)), pref, SLOT(setStatusFilterState(const bool)));
|
connect(statusLabel, SIGNAL(toggled(bool)), pref, SLOT(setStatusFilterState(const bool)));
|
||||||
connect(categoryLabel, SIGNAL(toggled(bool)), categoryFilters, SLOT(toggleFilter(bool)));
|
connect(trackerLabel, SIGNAL(toggled(bool)), m_trackerFilters, SLOT(toggleFilter(bool)));
|
||||||
connect(categoryLabel, SIGNAL(toggled(bool)), pref, SLOT(setCategoryFilterState(const bool)));
|
|
||||||
connect(trackerLabel, SIGNAL(toggled(bool)), trackerFilters, SLOT(toggleFilter(bool)));
|
|
||||||
connect(trackerLabel, SIGNAL(toggled(bool)), pref, SLOT(setTrackerFilterState(const bool)));
|
connect(trackerLabel, SIGNAL(toggled(bool)), pref, SLOT(setTrackerFilterState(const bool)));
|
||||||
connect(this, SIGNAL(trackerSuccess(const QString &, const QString &)), trackerFilters, SLOT(trackerSuccess(const QString &, const QString &)));
|
connect(this, SIGNAL(trackerSuccess(const QString &, const QString &)), m_trackerFilters, SLOT(trackerSuccess(const QString &, const QString &)));
|
||||||
connect(this, SIGNAL(trackerError(const QString &, const QString &)), trackerFilters, SLOT(trackerError(const QString &, const QString &)));
|
connect(this, SIGNAL(trackerError(const QString &, const QString &)), m_trackerFilters, SLOT(trackerError(const QString &, const QString &)));
|
||||||
connect(this, SIGNAL(trackerWarning(const QString &, const QString &)), trackerFilters, SLOT(trackerWarning(const QString &, const QString &)));
|
connect(this, SIGNAL(trackerWarning(const QString &, const QString &)), m_trackerFilters, SLOT(trackerWarning(const QString &, const QString &)));
|
||||||
}
|
}
|
||||||
|
|
||||||
void TransferListFiltersWidget::setDownloadTrackerFavicon(bool value)
|
void TransferListFiltersWidget::setDownloadTrackerFavicon(bool value)
|
||||||
{
|
{
|
||||||
trackerFilters->setDownloadTrackerFavicon(value);
|
m_trackerFilters->setDownloadTrackerFavicon(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TransferListFiltersWidget::addTrackers(BitTorrent::TorrentHandle *const torrent, const QList<BitTorrent::TrackerEntry> &trackers)
|
void TransferListFiltersWidget::addTrackers(BitTorrent::TorrentHandle *const torrent, const QList<BitTorrent::TrackerEntry> &trackers)
|
||||||
{
|
{
|
||||||
foreach (const BitTorrent::TrackerEntry &tracker, trackers)
|
foreach (const BitTorrent::TrackerEntry &tracker, trackers)
|
||||||
trackerFilters->addItem(tracker.url(), torrent->hash());
|
m_trackerFilters->addItem(tracker.url(), torrent->hash());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TransferListFiltersWidget::removeTrackers(BitTorrent::TorrentHandle *const torrent, const QList<BitTorrent::TrackerEntry> &trackers)
|
void TransferListFiltersWidget::removeTrackers(BitTorrent::TorrentHandle *const torrent, const QList<BitTorrent::TrackerEntry> &trackers)
|
||||||
{
|
{
|
||||||
foreach (const BitTorrent::TrackerEntry &tracker, trackers)
|
foreach (const BitTorrent::TrackerEntry &tracker, trackers)
|
||||||
trackerFilters->removeItem(tracker.url(), torrent->hash());
|
m_trackerFilters->removeItem(tracker.url(), torrent->hash());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TransferListFiltersWidget::changeTrackerless(BitTorrent::TorrentHandle *const torrent, bool trackerless)
|
void TransferListFiltersWidget::changeTrackerless(BitTorrent::TorrentHandle *const torrent, bool trackerless)
|
||||||
{
|
{
|
||||||
trackerFilters->changeTrackerless(trackerless, torrent->hash());
|
m_trackerFilters->changeTrackerless(trackerless, torrent->hash());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TransferListFiltersWidget::trackerSuccess(BitTorrent::TorrentHandle *const torrent, const QString &tracker)
|
void TransferListFiltersWidget::trackerSuccess(BitTorrent::TorrentHandle *const torrent, const QString &tracker)
|
||||||
|
@ -886,3 +636,9 @@ void TransferListFiltersWidget::trackerError(BitTorrent::TorrentHandle *const to
|
||||||
{
|
{
|
||||||
emit trackerError(torrent->hash(), tracker);
|
emit trackerError(torrent->hash(), tracker);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TransferListFiltersWidget::onCategoryFilterStateChanged(bool enabled)
|
||||||
|
{
|
||||||
|
m_categoryFilterWidget->setVisible(enabled);
|
||||||
|
m_transferList->applyCategoryFilter(enabled ? m_categoryFilterWidget->currentCategory() : QString());
|
||||||
|
}
|
||||||
|
|
|
@ -90,40 +90,6 @@ private:
|
||||||
virtual void torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const);
|
virtual void torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const);
|
||||||
};
|
};
|
||||||
|
|
||||||
class CategoryFiltersList: public FiltersBase
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
public:
|
|
||||||
CategoryFiltersList(QWidget *parent, TransferListWidget *transferList);
|
|
||||||
|
|
||||||
private slots:
|
|
||||||
// Redefine addItem() to make sure the list stays sorted
|
|
||||||
void addItem(const QString &category, bool hasTorrent = false);
|
|
||||||
void removeItem(const QString &category);
|
|
||||||
void removeSelectedCategory();
|
|
||||||
void removeUnusedCategories();
|
|
||||||
void torrentCategoryChanged(BitTorrent::TorrentHandle *const torrent, const QString &oldCategory);
|
|
||||||
void categoryRemoved(const QString &category);
|
|
||||||
void subcategoriesSupportChanged();
|
|
||||||
|
|
||||||
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(BitTorrent::TorrentHandle *const torrent);
|
|
||||||
virtual void torrentAboutToBeDeleted(BitTorrent::TorrentHandle *const torrent);
|
|
||||||
QString categoryFromRow(int row) const;
|
|
||||||
int rowFromCategory(const QString &category) const;
|
|
||||||
void refresh();
|
|
||||||
|
|
||||||
private:
|
|
||||||
QHash<QString, int> m_categories;
|
|
||||||
int m_totalTorrents;
|
|
||||||
int m_totalCategorized;
|
|
||||||
};
|
|
||||||
|
|
||||||
class TrackerFiltersList: public FiltersBase
|
class TrackerFiltersList: public FiltersBase
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
@ -169,6 +135,8 @@ private:
|
||||||
bool m_downloadTrackerFavicon;
|
bool m_downloadTrackerFavicon;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class CategoryFilterWidget;
|
||||||
|
|
||||||
class TransferListFiltersWidget: public QFrame
|
class TransferListFiltersWidget: public QFrame
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
@ -190,8 +158,13 @@ signals:
|
||||||
void trackerError(const QString &hash, const QString &tracker);
|
void trackerError(const QString &hash, const QString &tracker);
|
||||||
void trackerWarning(const QString &hash, const QString &tracker);
|
void trackerWarning(const QString &hash, const QString &tracker);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void onCategoryFilterStateChanged(bool enabled);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TrackerFiltersList *trackerFilters;
|
TransferListWidget *m_transferList;
|
||||||
|
TrackerFiltersList *m_trackerFilters;
|
||||||
|
CategoryFilterWidget *m_categoryFilterWidget;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // TRANSFERLISTFILTERSWIDGET_H
|
#endif // TRANSFERLISTFILTERSWIDGET_H
|
||||||
|
|
|
@ -7,6 +7,8 @@ CONFIG += c++11
|
||||||
DEFINES += BOOST_NO_CXX11_RVALUE_REFERENCES
|
DEFINES += BOOST_NO_CXX11_RVALUE_REFERENCES
|
||||||
greaterThan(QT_MAJOR_VERSION, 4): greaterThan(QT_MINOR_VERSION, 1): DEFINES += QBT_USES_QT5
|
greaterThan(QT_MAJOR_VERSION, 4): greaterThan(QT_MINOR_VERSION, 1): DEFINES += QBT_USES_QT5
|
||||||
|
|
||||||
|
lessThan(QT_MAJOR_VERSION, 5): DEFINES += QStringLiteral=QLatin1String
|
||||||
|
|
||||||
# Windows specific configuration
|
# Windows specific configuration
|
||||||
win32: include(../winconf.pri)
|
win32: include(../winconf.pri)
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue