qBittorrent/src/gui/transferlistmodel.cpp

769 lines
31 KiB
C++
Raw Normal View History

2010-11-14 22:32:29 +03: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 22:32:29 +03: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.
*/
#include "transferlistmodel.h"
2018-06-06 16:48:17 +03:00
#include <QApplication>
#include <QDateTime>
2018-06-06 16:48:17 +03:00
#include <QDebug>
#include "base/bittorrent/infohash.h"
2015-09-25 11:10:05 +03:00
#include "base/bittorrent/session.h"
#include "base/bittorrent/torrent.h"
#include "base/global.h"
#include "base/preferences.h"
#include "base/types.h"
#include "base/unicodestrings.h"
2015-09-25 11:10:05 +03:00
#include "base/utils/fs.h"
#include "base/utils/misc.h"
#include "base/utils/string.h"
#include "color.h"
#include "uithememanager.h"
#include "utils.h"
2022-05-26 22:12:10 +03:00
namespace
{
QColor getDefaultColorByState(const BitTorrent::TorrentState state)
{
const bool isDarkTheme = Utils::Gui::isDarkTheme();
2022-05-26 22:12:10 +03:00
switch (state)
{
case BitTorrent::TorrentState::Downloading:
case BitTorrent::TorrentState::ForcedDownloading:
case BitTorrent::TorrentState::DownloadingMetadata:
case BitTorrent::TorrentState::ForcedDownloadingMetadata:
return (isDarkTheme ? Color::Primer::Dark::successFg : Color::Primer::Light::successFg);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::StalledDownloading:
return (isDarkTheme ? Color::Primer::Dark::successEmphasis : Color::Primer::Light::successEmphasis);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::StalledUploading:
return (isDarkTheme ? Color::Primer::Dark::accentEmphasis : Color::Primer::Light::accentEmphasis);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::Uploading:
case BitTorrent::TorrentState::ForcedUploading:
return (isDarkTheme ? Color::Primer::Dark::accentFg : Color::Primer::Light::accentFg);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::PausedDownloading:
return (isDarkTheme ? Color::Primer::Dark::fgMuted : Color::Primer::Light::fgMuted);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::PausedUploading:
return (isDarkTheme ? Color::Primer::Dark::scaleBlue4 : Color::Primer::Light::scaleBlue4);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::QueuedDownloading:
case BitTorrent::TorrentState::QueuedUploading:
return (isDarkTheme ? Color::Primer::Dark::scaleYellow6 : Color::Primer::Light::scaleYellow6);
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::CheckingDownloading:
case BitTorrent::TorrentState::CheckingUploading:
case BitTorrent::TorrentState::CheckingResumeData:
case BitTorrent::TorrentState::Moving:
return (isDarkTheme ? Color::Primer::Dark::successFg : Color::Primer::Light::successFg);
case BitTorrent::TorrentState::Error:
case BitTorrent::TorrentState::MissingFiles:
2022-05-26 22:12:10 +03:00
case BitTorrent::TorrentState::Unknown:
return (isDarkTheme ? Color::Primer::Dark::dangerFg : Color::Primer::Light::dangerFg);
2022-05-26 22:12:10 +03:00
default:
Q_ASSERT(false);
break;
2022-05-26 22:12:10 +03:00
}
return {};
2022-05-26 22:12:10 +03:00
}
2014-05-11 15:29:06 +04:00
QHash<BitTorrent::TorrentState, QColor> torrentStateColorsFromUITheme()
{
struct TorrentStateColorDescriptor
{
const BitTorrent::TorrentState state;
const QString id;
};
const TorrentStateColorDescriptor colorDescriptors[] =
{
{BitTorrent::TorrentState::Downloading, u"TransferList.Downloading"_qs},
{BitTorrent::TorrentState::StalledDownloading, u"TransferList.StalledDownloading"_qs},
{BitTorrent::TorrentState::DownloadingMetadata, u"TransferList.DownloadingMetadata"_qs},
{BitTorrent::TorrentState::ForcedDownloadingMetadata, u"TransferList.ForcedDownloadingMetadata"_qs},
{BitTorrent::TorrentState::ForcedDownloading, u"TransferList.ForcedDownloading"_qs},
{BitTorrent::TorrentState::Uploading, u"TransferList.Uploading"_qs},
{BitTorrent::TorrentState::StalledUploading, u"TransferList.StalledUploading"_qs},
{BitTorrent::TorrentState::ForcedUploading, u"TransferList.ForcedUploading"_qs},
{BitTorrent::TorrentState::QueuedDownloading, u"TransferList.QueuedDownloading"_qs},
{BitTorrent::TorrentState::QueuedUploading, u"TransferList.QueuedUploading"_qs},
{BitTorrent::TorrentState::CheckingDownloading, u"TransferList.CheckingDownloading"_qs},
{BitTorrent::TorrentState::CheckingUploading, u"TransferList.CheckingUploading"_qs},
{BitTorrent::TorrentState::CheckingResumeData, u"TransferList.CheckingResumeData"_qs},
{BitTorrent::TorrentState::PausedDownloading, u"TransferList.PausedDownloading"_qs},
{BitTorrent::TorrentState::PausedUploading, u"TransferList.PausedUploading"_qs},
{BitTorrent::TorrentState::Moving, u"TransferList.Moving"_qs},
{BitTorrent::TorrentState::MissingFiles, u"TransferList.MissingFiles"_qs},
{BitTorrent::TorrentState::Error, u"TransferList.Error"_qs}
};
QHash<BitTorrent::TorrentState, QColor> colors;
2020-11-16 10:02:11 +03:00
for (const TorrentStateColorDescriptor &colorDescriptor : colorDescriptors)
{
const QColor themeColor = UIThemeManager::instance()->getColor(colorDescriptor.id, QColor());
if (themeColor.isValid())
colors.insert(colorDescriptor.state, themeColor);
}
return colors;
}
}
// TransferListModel
2014-05-11 15:29:06 +04:00
TransferListModel::TransferListModel(QObject *parent)
: QAbstractListModel {parent}
2020-11-16 10:02:11 +03:00
, m_statusStrings
{
{BitTorrent::TorrentState::Downloading, tr("Downloading")},
{BitTorrent::TorrentState::StalledDownloading, tr("Stalled", "Torrent is waiting for download to begin")},
{BitTorrent::TorrentState::DownloadingMetadata, tr("Downloading metadata", "Used when loading a magnet link")},
{BitTorrent::TorrentState::ForcedDownloadingMetadata, tr("[F] Downloading metadata", "Used when forced to load a magnet link. You probably shouldn't translate the F.")},
{BitTorrent::TorrentState::ForcedDownloading, tr("[F] Downloading", "Used when the torrent is forced started. You probably shouldn't translate the F.")},
{BitTorrent::TorrentState::Uploading, tr("Seeding", "Torrent is complete and in upload-only mode")},
{BitTorrent::TorrentState::StalledUploading, tr("Seeding", "Torrent is complete and in upload-only mode")},
{BitTorrent::TorrentState::ForcedUploading, tr("[F] Seeding", "Used when the torrent is forced started. You probably shouldn't translate the F.")},
{BitTorrent::TorrentState::QueuedDownloading, tr("Queued", "Torrent is queued")},
{BitTorrent::TorrentState::QueuedUploading, tr("Queued", "Torrent is queued")},
{BitTorrent::TorrentState::CheckingDownloading, tr("Checking", "Torrent local data is being checked")},
{BitTorrent::TorrentState::CheckingUploading, tr("Checking", "Torrent local data is being checked")},
{BitTorrent::TorrentState::CheckingResumeData, tr("Checking resume data", "Used when loading the torrents from disk after qbt is launched. It checks the correctness of the .fastresume file. Normally it is completed in a fraction of a second, unless loading many many torrents.")},
{BitTorrent::TorrentState::PausedDownloading, tr("Paused")},
{BitTorrent::TorrentState::PausedUploading, tr("Completed")},
{BitTorrent::TorrentState::Moving, tr("Moving", "Torrent local data are being moved/relocated")},
{BitTorrent::TorrentState::MissingFiles, tr("Missing Files")},
{BitTorrent::TorrentState::Error, tr("Errored", "Torrent status, the torrent has an error")}
}
, m_stateThemeColors {torrentStateColorsFromUITheme()}
, m_checkingIcon {UIThemeManager::instance()->getIcon(u"force-recheck"_qs)}
, m_completedIcon {UIThemeManager::instance()->getIcon(u"checked-completed"_qs)}
, m_downloadingIcon {UIThemeManager::instance()->getIcon(u"downloading"_qs)}
, m_errorIcon {UIThemeManager::instance()->getIcon(u"error"_qs)}
, m_pausedIcon {UIThemeManager::instance()->getIcon(u"torrent-stop"_qs)}
, m_queuedIcon {UIThemeManager::instance()->getIcon(u"queued"_qs)}
, m_stalledDLIcon {UIThemeManager::instance()->getIcon(u"stalledDL"_qs)}
, m_stalledUPIcon {UIThemeManager::instance()->getIcon(u"stalledUP"_qs)}
, m_uploadingIcon {UIThemeManager::instance()->getIcon(u"upload"_qs)}
2015-06-07 15:03:30 +03:00
{
configure();
connect(Preferences::instance(), &Preferences::changed, this, &TransferListModel::configure);
2015-06-07 15:03:30 +03:00
// Load the torrents
2018-04-18 16:59:41 +03:00
using namespace BitTorrent;
addTorrents(Session::instance()->torrents());
2014-05-11 15:29:06 +04:00
2015-06-07 15:03:30 +03:00
// Listen for torrent changes
connect(Session::instance(), &Session::torrentsLoaded, this, &TransferListModel::addTorrents);
connect(Session::instance(), &Session::torrentAboutToBeRemoved, this, &TransferListModel::handleTorrentAboutToBeRemoved);
connect(Session::instance(), &Session::torrentsUpdated, this, &TransferListModel::handleTorrentsUpdated);
connect(Session::instance(), &Session::torrentFinished, this, &TransferListModel::handleTorrentStatusUpdated);
connect(Session::instance(), &Session::torrentMetadataReceived, this, &TransferListModel::handleTorrentStatusUpdated);
connect(Session::instance(), &Session::torrentResumed, this, &TransferListModel::handleTorrentStatusUpdated);
connect(Session::instance(), &Session::torrentPaused, this, &TransferListModel::handleTorrentStatusUpdated);
connect(Session::instance(), &Session::torrentFinishedChecking, this, &TransferListModel::handleTorrentStatusUpdated);
2014-05-11 15:29:06 +04:00
}
int TransferListModel::rowCount(const QModelIndex &) const
{
return m_torrentList.size();
}
int TransferListModel::columnCount(const QModelIndex &) const
{
2015-06-07 15:03:30 +03:00
return NB_COLUMNS;
}
2022-06-22 10:47:13 +03:00
QVariant TransferListModel::headerData(const int section, const Qt::Orientation orientation, const int role) const
2015-04-19 18:17:47 +03:00
{
2020-11-16 10:02:11 +03:00
if (orientation == Qt::Horizontal)
{
if (role == Qt::DisplayRole)
{
switch (section)
{
case TR_QUEUE_POSITION: return QChar(u'#');
2015-06-07 15:03:30 +03:00
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("Progress", "% Done");
2015-06-07 15:03:30 +03:00
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");
case TR_TAGS: return tr("Tags");
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_DOWNLOAD_PATH: return tr("Incomplete Save Path", "Torrent incomplete save path");
2015-06-07 15:03:30 +03:00
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");
2019-07-21 10:20:54 +03:00
case TR_AVAILABILITY: return tr("Availability", "The number of distributed copies of the torrent");
case TR_INFOHASH_V1: return tr("Info Hash v1", "i.e: torrent info hash v1");
case TR_INFOHASH_V2: return tr("Info Hash v2", "i.e: torrent info hash v2");
2019-07-21 10:20:54 +03:00
default: return {};
2015-06-07 15:03:30 +03:00
}
}
2020-11-16 10:02:11 +03:00
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_QUEUE_POSITION:
2015-06-07 15:03:30 +03:00
case TR_LAST_ACTIVITY:
2019-07-21 10:20:54 +03:00
case TR_AVAILABILITY:
2019-02-22 04:59:31 +03:00
return QVariant(Qt::AlignRight | Qt::AlignVCenter);
2015-06-07 15:03:30 +03:00
default:
return QAbstractListModel::headerData(section, orientation, role);
}
}
}
2022-06-22 10:47:13 +03:00
return QAbstractListModel::headerData(section, orientation, role);
}
QString TransferListModel::displayValue(const BitTorrent::Torrent *torrent, const int column) const
{
bool hideValues = false;
if (m_hideZeroValuesMode == HideZeroValuesMode::Always)
hideValues = true;
2021-02-06 06:25:17 +03:00
else if (m_hideZeroValuesMode == HideZeroValuesMode::Paused)
hideValues = (torrent->state() == BitTorrent::TorrentState::PausedDownloading);
2015-04-19 18:17:47 +03:00
const auto availabilityString = [hideValues](const qreal value) -> QString
{
if (hideValues && (value == 0))
return {};
return (value >= 0)
? Utils::String::fromDouble(value, 3)
: tr("N/A");
};
const auto unitString = [hideValues](const qint64 value, const bool isSpeedUnit = false) -> QString
{
2020-10-11 20:22:27 +03:00
return (hideValues && (value == 0))
? QString {} : Utils::Misc::friendlyUnit(value, isSpeedUnit);
};
2015-04-19 18:17:47 +03:00
const auto limitString = [hideValues](const qint64 value) -> QString
{
if (hideValues && (value <= 0))
return {};
2015-04-19 18:17:47 +03:00
return (value > 0)
? Utils::Misc::friendlyUnit(value, true)
: C_INFINITY;
};
2015-04-19 18:17:47 +03:00
const auto amountString = [hideValues](const qint64 value, const qint64 total) -> QString
{
2021-02-06 06:25:17 +03:00
if (hideValues && (value == 0) && (total == 0))
return {};
return u"%1 (%2)"_qs.arg(QString::number(value), QString::number(total));
};
const auto etaString = [hideValues](const qlonglong value) -> QString
{
if (hideValues && (value >= MAX_ETA))
return {};
return Utils::Misc::userFriendlyDuration(value, MAX_ETA);
};
const auto ratioString = [hideValues](const qreal value) -> QString
{
2020-10-11 20:22:27 +03:00
if (hideValues && (value <= 0))
return {};
return ((static_cast<int>(value) == -1) || (value > BitTorrent::Torrent::MAX_RATIO))
? C_INFINITY : Utils::String::fromDouble(value, 2);
};
const auto queuePositionString = [](const qint64 value) -> QString
{
return (value >= 0) ? QString::number(value + 1) : u"*"_qs;
};
const auto lastActivityString = [hideValues](qint64 value) -> QString
{
if (hideValues && ((value < 0) || (value >= MAX_ETA)))
2021-02-06 06:25:17 +03:00
return {};
// Show '< 1m ago' when elapsed time is 0
if (value == 0)
value = 1;
return (value >= 0)
? tr("%1 ago", "e.g.: 1h 20m ago").arg(Utils::Misc::userFriendlyDuration(value))
: Utils::Misc::userFriendlyDuration(value);
};
const auto timeElapsedString = [hideValues](const qint64 elapsedTime, const qint64 seedingTime) -> QString
{
if (seedingTime <= 0)
{
if (hideValues && (elapsedTime == 0))
return {};
return Utils::Misc::userFriendlyDuration(elapsedTime);
}
return tr("%1 (seeded for %2)", "e.g. 4m39s (seeded for 3m10s)")
.arg(Utils::Misc::userFriendlyDuration(elapsedTime)
, Utils::Misc::userFriendlyDuration(seedingTime));
};
const auto progressString = [](const qreal progress) -> QString
{
return (progress >= 1)
? u"100%"_qs
: (Utils::String::fromDouble((progress * 100), 1) + u'%');
};
const auto statusString = [this](const BitTorrent::TorrentState state, const QString &errorMessage) -> QString
{
return (state == BitTorrent::TorrentState::Error)
? m_statusStrings[state] + u": " + errorMessage
: m_statusStrings[state];
};
const auto hashString = [hideValues](const auto &hash) -> QString
{
if (hideValues && !hash.isValid())
return {};
return hash.isValid() ? hash.toString() : tr("N/A");
};
2020-11-16 10:02:11 +03:00
switch (column)
{
case TR_NAME:
return torrent->name();
case TR_QUEUE_POSITION:
return queuePositionString(torrent->queuePosition());
case TR_SIZE:
return unitString(torrent->wantedSize());
case TR_PROGRESS:
return progressString(torrent->progress());
case TR_STATUS:
return statusString(torrent->state(), torrent->error());
case TR_SEEDS:
return amountString(torrent->seedsCount(), torrent->totalSeedsCount());
case TR_PEERS:
return amountString(torrent->leechsCount(), torrent->totalLeechersCount());
case TR_DLSPEED:
return unitString(torrent->downloadPayloadRate(), true);
case TR_UPSPEED:
return unitString(torrent->uploadPayloadRate(), true);
case TR_ETA:
return etaString(torrent->eta());
case TR_RATIO:
return ratioString(torrent->realRatio());
case TR_RATIO_LIMIT:
return ratioString(torrent->maxRatio());
case TR_CATEGORY:
return torrent->category();
case TR_TAGS:
return torrent->tags().join(u", "_qs);
case TR_ADD_DATE:
return QLocale().toString(torrent->addedTime().toLocalTime(), QLocale::ShortFormat);
case TR_SEED_DATE:
return QLocale().toString(torrent->completedTime().toLocalTime(), QLocale::ShortFormat);
case TR_TRACKER:
return torrent->currentTracker();
case TR_DLLIMIT:
return limitString(torrent->downloadLimit());
case TR_UPLIMIT:
return limitString(torrent->uploadLimit());
case TR_AMOUNT_DOWNLOADED:
return unitString(torrent->totalDownload());
case TR_AMOUNT_UPLOADED:
return unitString(torrent->totalUpload());
case TR_AMOUNT_DOWNLOADED_SESSION:
return unitString(torrent->totalPayloadDownload());
case TR_AMOUNT_UPLOADED_SESSION:
return unitString(torrent->totalPayloadUpload());
case TR_AMOUNT_LEFT:
return unitString(torrent->remainingSize());
case TR_TIME_ELAPSED:
return timeElapsedString(torrent->activeTime(), torrent->finishedTime());
case TR_SAVE_PATH:
return torrent->savePath().toString();
case TR_DOWNLOAD_PATH:
return torrent->downloadPath().toString();
case TR_COMPLETED:
return unitString(torrent->completedSize());
case TR_SEEN_COMPLETE_DATE:
return QLocale().toString(torrent->lastSeenComplete().toLocalTime(), QLocale::ShortFormat);
case TR_LAST_ACTIVITY:
return lastActivityString(torrent->timeSinceActivity());
case TR_AVAILABILITY:
return availabilityString(torrent->distributedCopies());
case TR_TOTAL_SIZE:
return unitString(torrent->totalSize());
case TR_INFOHASH_V1:
return hashString(torrent->infoHash().v1());
case TR_INFOHASH_V2:
return hashString(torrent->infoHash().v2());
}
return {};
}
QVariant TransferListModel::internalValue(const BitTorrent::Torrent *torrent, const int column, const bool alt) const
{
2020-11-16 10:02:11 +03:00
switch (column)
{
case TR_NAME:
2015-06-07 15:03:30 +03:00
return torrent->name();
case TR_QUEUE_POSITION:
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:
return torrent->progress() * 100;
case TR_STATUS:
return QVariant::fromValue(torrent->state());
2015-04-19 18:17:47 +03:00
case TR_SEEDS:
return !alt ? torrent->seedsCount() : torrent->totalSeedsCount();
2015-04-19 18:17:47 +03:00
case TR_PEERS:
return !alt ? 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_TAGS:
return QVariant::fromValue(torrent->tags());
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:
return torrent->remainingSize();
case TR_TIME_ELAPSED:
return !alt ? torrent->activeTime() : torrent->finishedTime();
case TR_DOWNLOAD_PATH:
return torrent->downloadPath().data();
case TR_SAVE_PATH:
return torrent->savePath().data();
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:
return torrent->timeSinceActivity();
2019-07-21 10:20:54 +03:00
case TR_AVAILABILITY:
return torrent->distributedCopies();
case TR_TOTAL_SIZE:
return torrent->totalSize();
case TR_INFOHASH_V1:
return QVariant::fromValue(torrent->infoHash().v1());
case TR_INFOHASH_V2:
return QVariant::fromValue(torrent->infoHash().v2());
}
2019-02-14 20:16:42 +03:00
return {};
}
QVariant TransferListModel::data(const QModelIndex &index, const int role) const
{
if (!index.isValid()) return {};
const BitTorrent::Torrent *torrent = m_torrentList.value(index.row());
if (!torrent) return {};
2020-11-16 10:02:11 +03:00
switch (role)
{
case Qt::ForegroundRole:
return m_stateThemeColors.value(torrent->state(), getDefaultColorByState(torrent->state()));
case Qt::DisplayRole:
return displayValue(torrent, index.column());
case UnderlyingDataRole:
2021-03-18 07:04:46 +03:00
return internalValue(torrent, index.column(), false);
case AdditionalUnderlyingDataRole:
return internalValue(torrent, index.column(), true);
case Qt::DecorationRole:
if (index.column() == TR_NAME)
return getIconByState(torrent->state());
break;
case Qt::ToolTipRole:
2020-11-16 10:02:11 +03:00
switch (index.column())
{
case TR_NAME:
case TR_STATUS:
case TR_CATEGORY:
case TR_TAGS:
case TR_TRACKER:
case TR_SAVE_PATH:
case TR_DOWNLOAD_PATH:
case TR_INFOHASH_V1:
case TR_INFOHASH_V2:
return displayValue(torrent, index.column());
}
break;
case Qt::TextAlignmentRole:
2020-11-16 10:02:11 +03:00
switch (index.column())
{
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_QUEUE_POSITION:
case TR_LAST_ACTIVITY:
case TR_AVAILABILITY:
2022-06-22 10:47:13 +03:00
return QVariant(Qt::AlignRight | Qt::AlignVCenter);
}
2022-06-22 10:47:13 +03:00
break;
default:
break;
}
return {};
}
bool TransferListModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
2015-06-07 15:03:30 +03:00
if (!index.isValid() || (role != Qt::DisplayRole)) return false;
BitTorrent::Torrent *const torrent = m_torrentList.value(index.row());
2015-06-07 15:03:30 +03:00
if (!torrent) return false;
2015-04-19 18:17:47 +03:00
2019-08-07 09:04:03 +03:00
// Category and Name columns can be edited
2020-11-16 10:02:11 +03:00
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;
}
void TransferListModel::addTorrents(const QVector<BitTorrent::Torrent *> &torrents)
2015-06-07 15:03:30 +03:00
{
qsizetype row = m_torrentList.size();
const qsizetype total = row + torrents.size();
beginInsertRows({}, row, total);
m_torrentList.reserve(total);
for (BitTorrent::Torrent *torrent : torrents)
{
Q_ASSERT(!m_torrentMap.contains(torrent));
m_torrentList.append(torrent);
m_torrentMap[torrent] = row++;
}
endInsertRows();
}
Qt::ItemFlags TransferListModel::flags(const QModelIndex &index) const
{
2018-09-07 14:12:38 +03:00
if (!index.isValid()) return Qt::NoItemFlags;
2015-06-07 15:03:30 +03:00
// Explicitly mark as editable
return QAbstractListModel::flags(index) | Qt::ItemIsEditable;
}
BitTorrent::Torrent *TransferListModel::torrentHandle(const QModelIndex &index) const
{
2018-09-07 14:12:38 +03:00
if (!index.isValid()) return nullptr;
2015-04-19 18:17:47 +03:00
return m_torrentList.value(index.row());
}
void TransferListModel::handleTorrentAboutToBeRemoved(BitTorrent::Torrent *const torrent)
{
const int row = m_torrentMap.value(torrent, -1);
2019-08-28 15:43:02 +03:00
Q_ASSERT(row >= 0);
beginRemoveRows({}, row, row);
m_torrentList.removeAt(row);
m_torrentMap.remove(torrent);
2020-11-16 10:02:11 +03:00
for (int &value : m_torrentMap)
{
if (value > row)
--value;
2015-04-19 18:17:47 +03:00
}
endRemoveRows();
}
void TransferListModel::handleTorrentStatusUpdated(BitTorrent::Torrent *const torrent)
{
const int row = m_torrentMap.value(torrent, -1);
2019-08-28 15:43:02 +03:00
Q_ASSERT(row >= 0);
emit dataChanged(index(row, 0), index(row, columnCount() - 1));
}
void TransferListModel::handleTorrentsUpdated(const QVector<BitTorrent::Torrent *> &torrents)
{
const int columns = (columnCount() - 1);
2020-11-16 10:02:11 +03:00
if (torrents.size() <= (m_torrentList.size() * 0.5))
{
for (BitTorrent::Torrent *const torrent : torrents)
2020-11-16 10:02:11 +03:00
{
2019-08-28 10:32:21 +03:00
const int row = m_torrentMap.value(torrent, -1);
2019-08-28 15:43:02 +03:00
Q_ASSERT(row >= 0);
2019-08-28 10:32:21 +03:00
emit dataChanged(index(row, 0), index(row, columns));
}
}
2020-11-16 10:02:11 +03:00
else
{
2019-08-28 10:32:21 +03:00
// save the overhead when more than half of the torrent list needs update
emit dataChanged(index(0, 0), index((rowCount() - 1), columns));
}
}
void TransferListModel::configure()
{
const Preferences *pref = Preferences::instance();
HideZeroValuesMode hideZeroValuesMode = HideZeroValuesMode::Never;
2020-11-16 10:02:11 +03:00
if (pref->getHideZeroValues())
{
if (pref->getHideZeroComboValues() == 1)
hideZeroValuesMode = HideZeroValuesMode::Paused;
else
hideZeroValuesMode = HideZeroValuesMode::Always;
}
2020-11-16 10:02:11 +03:00
if (m_hideZeroValuesMode != hideZeroValuesMode)
{
m_hideZeroValuesMode = hideZeroValuesMode;
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
}
}
QIcon TransferListModel::getIconByState(const BitTorrent::TorrentState state) const
{
switch (state)
{
case BitTorrent::TorrentState::Downloading:
case BitTorrent::TorrentState::ForcedDownloading:
case BitTorrent::TorrentState::DownloadingMetadata:
case BitTorrent::TorrentState::ForcedDownloadingMetadata:
return m_downloadingIcon;
case BitTorrent::TorrentState::StalledDownloading:
return m_stalledDLIcon;
case BitTorrent::TorrentState::StalledUploading:
return m_stalledUPIcon;
case BitTorrent::TorrentState::Uploading:
case BitTorrent::TorrentState::ForcedUploading:
return m_uploadingIcon;
case BitTorrent::TorrentState::PausedDownloading:
return m_pausedIcon;
case BitTorrent::TorrentState::PausedUploading:
return m_completedIcon;
case BitTorrent::TorrentState::QueuedDownloading:
case BitTorrent::TorrentState::QueuedUploading:
return m_queuedIcon;
case BitTorrent::TorrentState::CheckingDownloading:
case BitTorrent::TorrentState::CheckingUploading:
case BitTorrent::TorrentState::CheckingResumeData:
case BitTorrent::TorrentState::Moving:
return m_checkingIcon;
case BitTorrent::TorrentState::Unknown:
case BitTorrent::TorrentState::MissingFiles:
case BitTorrent::TorrentState::Error:
return m_errorIcon;
default:
Q_ASSERT(false);
return m_errorIcon;
}
}