Vladimir Golovnev (Glassez)
8 years ago
8 changed files with 912 additions and 327 deletions
@ -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; |
||||||
|
} |
@ -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
|
@ -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(); |
||||||
|
} |
@ -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(); |
||||||
|
}; |
Loading…
Reference in new issue