qBittorrent/src/gui/torrentmodel.cpp

473 lines
17 KiB
C++
Raw Normal View History

2010-11-14 19:32:29 +00:00
/*
2015-06-07 15:03:30 +03:00
* Bittorrent Client using Qt and libtorrent.
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru>
* Copyright (C) 2010 Christophe Dumez <chris@qbittorrent.org>
2010-11-14 19:32:29 +00:00
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* In addition, as a special exception, the copyright holders give permission to
* link this program with the OpenSSL project's "OpenSSL" library (or with
* modified versions of it that use the same license as the "OpenSSL" library),
* and distribute the linked executables. You must obey the GNU General Public
* License in all respects for all of the code used other than "OpenSSL". If you
* modify file(s), you may extend this exception to your version of the file(s),
* but you are not obligated to do so. If you do not wish to do so, delete this
* exception statement from your version.
*
* Contact : chris@qbittorrent.org
*/
#include <QDebug>
#include <QApplication>
#include <QPalette>
2015-04-19 18:17:47 +03:00
#include <QIcon>
2015-09-25 11:10:05 +03:00
#include "base/bittorrent/session.h"
#include "base/bittorrent/torrenthandle.h"
#include "base/torrentfilter.h"
#include "base/utils/fs.h"
2015-04-19 18:17:47 +03:00
#include "torrentmodel.h"
2015-06-07 15:03:30 +03:00
static QIcon getIconByState(BitTorrent::TorrentState state);
static QColor getColorByState(BitTorrent::TorrentState state);
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
static QIcon getPausedIcon();
static QIcon getQueuedIcon();
static QIcon getDownloadingIcon();
static QIcon getStalledDownloadingIcon();
static QIcon getUploadingIcon();
static QIcon getStalledUploadingIcon();
static QIcon getCompletedIcon();
static QIcon getCheckingIcon();
static QIcon getErrorIcon();
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
static bool isDarkTheme();
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
// TorrentModel
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
TorrentModel::TorrentModel(QObject *parent)
: QAbstractListModel(parent)
{
// Load the torrents
foreach (BitTorrent::TorrentHandle *const torrent, BitTorrent::Session::instance()->torrents())
addTorrent(torrent);
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
// Listen for torrent changes
connect(BitTorrent::Session::instance(), SIGNAL(torrentAdded(BitTorrent::TorrentHandle * const)), SLOT(addTorrent(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentAboutToBeRemoved(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentAboutToBeRemoved(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentsUpdated()), SLOT(handleTorrentsUpdated()));
connect(BitTorrent::Session::instance(), SIGNAL(torrentFinished(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentStatusUpdated(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentMetadataLoaded(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentStatusUpdated(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentResumed(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentStatusUpdated(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentPaused(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentStatusUpdated(BitTorrent::TorrentHandle * const)));
connect(BitTorrent::Session::instance(), SIGNAL(torrentFinishedChecking(BitTorrent::TorrentHandle * const)), SLOT(handleTorrentStatusUpdated(BitTorrent::TorrentHandle * const)));
2014-05-11 15:29:06 +04:00
}
2015-06-07 15:03:30 +03:00
int TorrentModel::rowCount(const QModelIndex &index) const
{
2015-06-07 15:03:30 +03:00
Q_UNUSED(index);
return m_torrents.size();
}
2015-06-07 15:03:30 +03:00
int TorrentModel::columnCount(const QModelIndex &parent) const
{
2015-06-07 15:03:30 +03:00
Q_UNUSED(parent);
return NB_COLUMNS;
}
2015-06-07 15:03:30 +03:00
QVariant TorrentModel::headerData(int section, Qt::Orientation orientation, int role) const
2015-04-19 18:17:47 +03:00
{
2015-06-07 15:03:30 +03:00
if (orientation == Qt::Horizontal) {
if (role == Qt::DisplayRole) {
switch (section) {
2015-06-07 15:03:30 +03:00
case TR_PRIORITY: return "#";
case TR_NAME: return tr("Name", "i.e: torrent name");
case TR_SIZE: return tr("Size", "i.e: torrent size");
case TR_PROGRESS: return tr("Done", "% Done");
case TR_STATUS: return tr("Status", "Torrent status (e.g. downloading, seeding, paused)");
case TR_SEEDS: return tr("Seeds", "i.e. full sources (often untranslated)");
case TR_PEERS: return tr("Peers", "i.e. partial sources (often untranslated)");
case TR_DLSPEED: return tr("Down Speed", "i.e: Download speed");
case TR_UPSPEED: return tr("Up Speed", "i.e: Upload speed");
case TR_RATIO: return tr("Ratio", "Share ratio");
case TR_ETA: return tr("ETA", "i.e: Estimated Time of Arrival / Time left");
case TR_CATEGORY: return tr("Category");
2015-06-07 15:03:30 +03:00
case TR_ADD_DATE: return tr("Added On", "Torrent was added to transfer list on 01/01/2010 08:00");
case TR_SEED_DATE: return tr("Completed On", "Torrent was completed on 01/01/2010 08:00");
case TR_TRACKER: return tr("Tracker");
case TR_DLLIMIT: return tr("Down Limit", "i.e: Download limit");
case TR_UPLIMIT: return tr("Up Limit", "i.e: Upload limit");
case TR_AMOUNT_DOWNLOADED: return tr("Downloaded", "Amount of data downloaded (e.g. in MB)");
case TR_AMOUNT_UPLOADED: return tr("Uploaded", "Amount of data uploaded (e.g. in MB)");
case TR_AMOUNT_DOWNLOADED_SESSION: return tr("Session Download", "Amount of data downloaded since program open (e.g. in MB)");
case TR_AMOUNT_UPLOADED_SESSION: return tr("Session Upload", "Amount of data uploaded since program open (e.g. in MB)");
case TR_AMOUNT_LEFT: return tr("Remaining", "Amount of data left to download (e.g. in MB)");
case TR_TIME_ELAPSED: return tr("Time Active", "Time (duration) the torrent is active (not paused)");
case TR_SAVE_PATH: return tr("Save path", "Torrent save path");
case TR_COMPLETED: return tr("Completed", "Amount of data completed (e.g. in MB)");
case TR_RATIO_LIMIT: return tr("Ratio Limit", "Upload share ratio limit");
case TR_SEEN_COMPLETE_DATE: return tr("Last Seen Complete", "Indicates the time when the torrent was last seen complete/whole");
case TR_LAST_ACTIVITY: return tr("Last Activity", "Time passed since a chunk was downloaded/uploaded");
case TR_TOTAL_SIZE: return tr("Total Size", "i.e. Size including unwanted data");
default:
return QVariant();
}
}
else if (role == Qt::TextAlignmentRole) {
switch (section) {
2015-06-07 15:03:30 +03:00
case TR_AMOUNT_DOWNLOADED:
case TR_AMOUNT_UPLOADED:
case TR_AMOUNT_DOWNLOADED_SESSION:
case TR_AMOUNT_UPLOADED_SESSION:
case TR_AMOUNT_LEFT:
case TR_COMPLETED:
case TR_SIZE:
case TR_TOTAL_SIZE:
case TR_ETA:
case TR_SEEDS:
case TR_PEERS:
case TR_UPSPEED:
case TR_DLSPEED:
case TR_UPLIMIT:
case TR_DLLIMIT:
case TR_RATIO_LIMIT:
case TR_RATIO:
case TR_PRIORITY:
case TR_LAST_ACTIVITY:
return QVariant(Qt::AlignRight | Qt::AlignVCenter);
default:
return QAbstractListModel::headerData(section, orientation, role);
}
}
}
2015-06-07 15:03:30 +03:00
return QVariant();
}
2015-06-07 15:03:30 +03:00
QVariant TorrentModel::data(const QModelIndex &index, int role) const
{
2015-06-07 15:03:30 +03:00
if (!index.isValid()) return QVariant();
2015-04-19 18:17:47 +03:00
2015-06-07 15:03:30 +03:00
BitTorrent::TorrentHandle *const torrent = m_torrents.value(index.row());
if (!torrent) return QVariant();
2015-06-07 15:03:30 +03:00
if ((role == Qt::DecorationRole) && (index.column() == TR_NAME))
return getIconByState(torrent->state());
2015-04-19 18:17:47 +03:00
if (role == Qt::ForegroundRole)
2015-06-07 15:03:30 +03:00
return getColorByState(torrent->state());
2015-04-19 18:17:47 +03:00
if ((role != Qt::DisplayRole) && (role != Qt::UserRole))
return QVariant();
switch (index.column()) {
case TR_NAME:
2015-06-07 15:03:30 +03:00
return torrent->name();
2015-04-19 18:17:47 +03:00
case TR_PRIORITY:
2015-06-07 15:03:30 +03:00
return torrent->queuePosition();
case TR_SIZE:
2015-06-07 15:03:30 +03:00
return torrent->wantedSize();
case TR_PROGRESS:
2015-06-07 15:03:30 +03:00
return torrent->progress();
case TR_STATUS:
2015-06-07 15:03:30 +03:00
return static_cast<int>(torrent->state());
2015-04-19 18:17:47 +03:00
case TR_SEEDS:
2016-02-01 20:07:18 +08:00
return (role == Qt::DisplayRole) ? torrent->seedsCount() : torrent->totalSeedsCount();
2015-04-19 18:17:47 +03:00
case TR_PEERS:
2016-02-01 20:07:18 +08:00
return (role == Qt::DisplayRole) ? torrent->leechsCount() : torrent->totalLeechersCount();
case TR_DLSPEED:
2015-06-07 15:03:30 +03:00
return torrent->downloadPayloadRate();
case TR_UPSPEED:
2015-06-07 15:03:30 +03:00
return torrent->uploadPayloadRate();
2015-04-19 18:17:47 +03:00
case TR_ETA:
2015-06-07 15:03:30 +03:00
return torrent->eta();
case TR_RATIO:
2015-06-07 15:03:30 +03:00
return torrent->realRatio();
case TR_CATEGORY:
return torrent->category();
case TR_ADD_DATE:
2015-06-07 15:03:30 +03:00
return torrent->addedTime();
case TR_SEED_DATE:
2015-06-07 15:03:30 +03:00
return torrent->completedTime();
case TR_TRACKER:
2015-06-07 15:03:30 +03:00
return torrent->currentTracker();
case TR_DLLIMIT:
2015-06-07 15:03:30 +03:00
return torrent->downloadLimit();
case TR_UPLIMIT:
2015-06-07 15:03:30 +03:00
return torrent->uploadLimit();
case TR_AMOUNT_DOWNLOADED:
2015-06-07 15:03:30 +03:00
return torrent->totalDownload();
case TR_AMOUNT_UPLOADED:
2015-06-07 15:03:30 +03:00
return torrent->totalUpload();
case TR_AMOUNT_DOWNLOADED_SESSION:
2015-06-07 15:03:30 +03:00
return torrent->totalPayloadDownload();
case TR_AMOUNT_UPLOADED_SESSION:
2015-06-07 15:03:30 +03:00
return torrent->totalPayloadUpload();
case TR_AMOUNT_LEFT:
2015-06-07 15:03:30 +03:00
return torrent->incompletedSize();
case TR_TIME_ELAPSED:
2015-06-07 15:03:30 +03:00
return (role == Qt::DisplayRole) ? torrent->activeTime() : torrent->seedingTime();
case TR_SAVE_PATH:
return Utils::Fs::toNativePath(torrent->savePath());
case TR_COMPLETED:
2015-06-07 15:03:30 +03:00
return torrent->completedSize();
2015-04-19 18:17:47 +03:00
case TR_RATIO_LIMIT:
2015-06-07 15:03:30 +03:00
return torrent->maxRatio();
case TR_SEEN_COMPLETE_DATE:
2015-06-07 15:03:30 +03:00
return torrent->lastSeenComplete();
case TR_LAST_ACTIVITY:
2015-06-07 15:03:30 +03:00
if (torrent->isPaused() || torrent->isChecking())
return -1;
2015-06-07 15:03:30 +03:00
return torrent->timeSinceActivity();
case TR_TOTAL_SIZE:
return torrent->totalSize();
default:
return QVariant();
}
2015-06-07 15:03:30 +03:00
return QVariant();
}
2015-06-07 15:03:30 +03:00
bool TorrentModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
2015-06-07 15:03:30 +03:00
qDebug() << Q_FUNC_INFO << value;
if (!index.isValid() || (role != Qt::DisplayRole)) return false;
2015-06-07 15:03:30 +03:00
qDebug("Index is valid and role is DisplayRole");
BitTorrent::TorrentHandle *const torrent = m_torrents.value(index.row());
if (!torrent) return false;
2015-04-19 18:17:47 +03:00
// Category, seed date and Name columns can be edited
switch (index.column()) {
2015-06-07 15:03:30 +03:00
case TR_NAME:
torrent->setName(value.toString());
break;
case TR_CATEGORY:
torrent->setCategory(value.toString());
2015-06-07 15:03:30 +03:00
break;
default:
return false;
}
2015-04-19 18:17:47 +03:00
2015-06-07 15:03:30 +03:00
return true;
}
2015-06-07 15:03:30 +03:00
void TorrentModel::addTorrent(BitTorrent::TorrentHandle *const torrent)
{
if (m_torrents.indexOf(torrent) == -1) {
const int row = m_torrents.size();
beginInsertRows(QModelIndex(), row, row);
m_torrents << torrent;
endInsertRows();
}
}
2015-06-07 15:03:30 +03:00
Qt::ItemFlags TorrentModel::flags(const QModelIndex &index) const
{
2015-06-07 15:03:30 +03:00
if (!index.isValid()) return 0;
2015-06-07 15:03:30 +03:00
// Explicitly mark as editable
return QAbstractListModel::flags(index) | Qt::ItemIsEditable;
}
2015-06-07 15:03:30 +03:00
BitTorrent::TorrentHandle *TorrentModel::torrentHandle(const QModelIndex &index) const
{
2015-06-07 15:03:30 +03:00
if (!index.isValid()) return 0;
2015-04-19 18:17:47 +03:00
2015-06-07 15:03:30 +03:00
return m_torrents.value(index.row());
}
2015-06-07 15:03:30 +03:00
void TorrentModel::handleTorrentAboutToBeRemoved(BitTorrent::TorrentHandle *const torrent)
{
2015-06-07 15:03:30 +03:00
const int row = m_torrents.indexOf(torrent);
if (row >= 0) {
beginRemoveRows(QModelIndex(), row, row);
m_torrents.removeAt(row);
endRemoveRows();
2015-04-19 18:17:47 +03:00
}
}
2015-06-07 15:03:30 +03:00
void TorrentModel::handleTorrentStatusUpdated(BitTorrent::TorrentHandle *const torrent)
{
2015-06-07 15:03:30 +03:00
const int row = m_torrents.indexOf(torrent);
if (row >= 0)
emit dataChanged(index(row, 0), index(row, columnCount() - 1));
}
void TorrentModel::handleTorrentsUpdated()
{
emit dataChanged(index(0, 0), index(rowCount() - 1, columnCount() - 1));
}
2015-06-07 15:03:30 +03:00
// Static functions
QIcon getIconByState(BitTorrent::TorrentState state)
{
2015-06-07 15:03:30 +03:00
switch (state) {
case BitTorrent::TorrentState::Downloading:
case BitTorrent::TorrentState::ForcedDownloading:
case BitTorrent::TorrentState::DownloadingMetadata:
return getDownloadingIcon();
case BitTorrent::TorrentState::Allocating:
case BitTorrent::TorrentState::StalledDownloading:
return getStalledDownloadingIcon();
case BitTorrent::TorrentState::StalledUploading:
return getStalledUploadingIcon();
case BitTorrent::TorrentState::Uploading:
case BitTorrent::TorrentState::ForcedUploading:
return getUploadingIcon();
case BitTorrent::TorrentState::PausedDownloading:
return getPausedIcon();
case BitTorrent::TorrentState::PausedUploading:
return getCompletedIcon();
case BitTorrent::TorrentState::QueuedDownloading:
case BitTorrent::TorrentState::QueuedUploading:
return getQueuedIcon();
case BitTorrent::TorrentState::CheckingDownloading:
case BitTorrent::TorrentState::CheckingUploading:
#if LIBTORRENT_VERSION_NUM < 10100
2015-06-30 11:03:46 +03:00
case BitTorrent::TorrentState::QueuedForChecking:
#endif
2015-06-30 11:03:46 +03:00
case BitTorrent::TorrentState::CheckingResumeData:
2015-06-07 15:03:30 +03:00
return getCheckingIcon();
case BitTorrent::TorrentState::Unknown:
case BitTorrent::TorrentState::MissingFiles:
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::Error:
return getErrorIcon();
default:
Q_ASSERT(false);
return getErrorIcon();
}
}
2015-06-07 15:03:30 +03:00
QColor getColorByState(BitTorrent::TorrentState state)
{
// Color names taken from http://cloford.com/resources/colours/500col.htm
2015-06-07 15:03:30 +03:00
bool dark = isDarkTheme();
switch (state) {
case BitTorrent::TorrentState::Downloading:
case BitTorrent::TorrentState::ForcedDownloading:
case BitTorrent::TorrentState::DownloadingMetadata:
if (!dark)
return QColor(34, 139, 34); // Forest Green
else
return QColor(50, 205, 50); // Lime Green
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::Allocating:
case BitTorrent::TorrentState::StalledDownloading:
case BitTorrent::TorrentState::StalledUploading:
if (!dark)
return QColor(0, 0, 0); // Black
else
return QColor(204, 204, 204); // Gray 80
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::Uploading:
case BitTorrent::TorrentState::ForcedUploading:
if (!dark)
return QColor(65, 105, 225); // Royal Blue
else
return QColor(99, 184, 255); // Steel Blue 1
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::PausedDownloading:
return QColor(250, 128, 114); // Salmon
case BitTorrent::TorrentState::PausedUploading:
if (!dark)
return QColor(0, 0, 139); // Dark Blue
else
return QColor(79, 148, 205); // Steel Blue 3
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::Error:
case BitTorrent::TorrentState::MissingFiles:
2015-06-07 15:03:30 +03:00
return QColor(255, 0, 0); // red
case BitTorrent::TorrentState::QueuedDownloading:
case BitTorrent::TorrentState::QueuedUploading:
case BitTorrent::TorrentState::CheckingDownloading:
case BitTorrent::TorrentState::CheckingUploading:
#if LIBTORRENT_VERSION_NUM < 10100
2015-06-30 11:03:46 +03:00
case BitTorrent::TorrentState::QueuedForChecking:
#endif
2015-06-30 11:03:46 +03:00
case BitTorrent::TorrentState::CheckingResumeData:
if (!dark)
return QColor(0, 128, 128); // Teal
else
return QColor(0, 205, 205); // Cyan 3
2015-06-07 15:03:30 +03:00
case BitTorrent::TorrentState::Unknown:
return QColor(255, 0, 0); // red
default:
Q_ASSERT(false);
return QColor(255, 0, 0); // red
}
}
2015-06-07 15:03:30 +03:00
QIcon getPausedIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/paused.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getQueuedIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/queued.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getDownloadingIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/downloading.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getStalledDownloadingIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/stalledDL.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getUploadingIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/uploading.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getStalledUploadingIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/stalledUP.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getCompletedIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/completed.png");
return cached;
}
2015-06-07 15:03:30 +03:00
QIcon getCheckingIcon()
2015-04-19 18:17:47 +03:00
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/checking.png");
return cached;
2015-04-19 18:17:47 +03:00
}
2015-06-07 15:03:30 +03:00
QIcon getErrorIcon()
{
2015-06-07 15:03:30 +03:00
static QIcon cached = QIcon(":/icons/skin/error.png");
return cached;
}
2015-06-07 15:03:30 +03:00
bool isDarkTheme()
{
2015-06-07 15:03:30 +03:00
QPalette pal = QApplication::palette();
// QPalette::Base is used for the background of the Treeview
QColor color = pal.color(QPalette::Active, QPalette::Base);
return (color.lightness() < 127);
}