/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2015 Vladimir Golovnev * Copyright (C) 2010 Christophe Dumez * * 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" #include #include #include #include #include "base/bittorrent/infohash.h" #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" #include "base/utils/fs.h" #include "base/utils/misc.h" #include "base/utils/string.h" #include "uithememanager.h" namespace { QColor getDefaultColorByState(const BitTorrent::TorrentState state) { switch (state) { case BitTorrent::TorrentState::Downloading: case BitTorrent::TorrentState::ForcedDownloading: case BitTorrent::TorrentState::DownloadingMetadata: case BitTorrent::TorrentState::ForcedDownloadingMetadata: return QColorConstants::Svg::green; case BitTorrent::TorrentState::StalledDownloading: return QColorConstants::Svg::mediumseagreen; case BitTorrent::TorrentState::StalledUploading: return QColorConstants::Svg::cornflowerblue; case BitTorrent::TorrentState::Uploading: case BitTorrent::TorrentState::ForcedUploading: return QColorConstants::Svg::royalblue; case BitTorrent::TorrentState::PausedDownloading: return QColorConstants::Svg::grey; case BitTorrent::TorrentState::PausedUploading: return QColorConstants::Svg::darkslateblue; case BitTorrent::TorrentState::QueuedDownloading: case BitTorrent::TorrentState::QueuedUploading: return QColorConstants::Svg::peru; case BitTorrent::TorrentState::CheckingDownloading: case BitTorrent::TorrentState::CheckingUploading: case BitTorrent::TorrentState::CheckingResumeData: case BitTorrent::TorrentState::Moving: return QColorConstants::Svg::teal; case BitTorrent::TorrentState::Error: case BitTorrent::TorrentState::MissingFiles: case BitTorrent::TorrentState::Unknown: return QColorConstants::Svg::red; default: Q_ASSERT(false); return QColorConstants::Svg::red; } } QHash 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 colors; 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 TransferListModel::TransferListModel(QObject *parent) : QAbstractListModel {parent} , 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)} { configure(); connect(Preferences::instance(), &Preferences::changed, this, &TransferListModel::configure); // Load the torrents using namespace BitTorrent; addTorrents(Session::instance()->torrents()); // 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); } int TransferListModel::rowCount(const QModelIndex &) const { return m_torrentList.size(); } int TransferListModel::columnCount(const QModelIndex &) const { return NB_COLUMNS; } QVariant TransferListModel::headerData(const int section, const Qt::Orientation orientation, const int role) const { if (orientation == Qt::Horizontal) { if (role == Qt::DisplayRole) { switch (section) { case TR_QUEUE_POSITION: return QChar(u'#'); 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"); 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"); 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"); 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"); 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"); default: return {}; } } else if (role == Qt::TextAlignmentRole) { switch (section) { 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: return QVariant(Qt::AlignRight | Qt::AlignVCenter); default: return QAbstractListModel::headerData(section, orientation, role); } } } 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; else if (m_hideZeroValuesMode == HideZeroValuesMode::Paused) hideValues = (torrent->state() == BitTorrent::TorrentState::PausedDownloading); 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 { return (hideValues && (value == 0)) ? QString {} : Utils::Misc::friendlyUnit(value, isSpeedUnit); }; const auto limitString = [hideValues](const qint64 value) -> QString { if (hideValues && (value <= 0)) return {}; return (value > 0) ? Utils::Misc::friendlyUnit(value, true) : C_INFINITY; }; const auto amountString = [hideValues](const qint64 value, const qint64 total) -> QString { 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 { if (hideValues && (value <= 0)) return {}; return ((static_cast(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))) 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"); }; 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 { switch (column) { case TR_NAME: return torrent->name(); case TR_QUEUE_POSITION: return torrent->queuePosition(); case TR_SIZE: return torrent->wantedSize(); case TR_PROGRESS: return torrent->progress() * 100; case TR_STATUS: return QVariant::fromValue(torrent->state()); case TR_SEEDS: return !alt ? torrent->seedsCount() : torrent->totalSeedsCount(); case TR_PEERS: return !alt ? torrent->leechsCount() : torrent->totalLeechersCount(); case TR_DLSPEED: return torrent->downloadPayloadRate(); case TR_UPSPEED: return torrent->uploadPayloadRate(); case TR_ETA: return torrent->eta(); case TR_RATIO: return torrent->realRatio(); case TR_CATEGORY: return torrent->category(); case TR_TAGS: return QVariant::fromValue(torrent->tags()); case TR_ADD_DATE: return torrent->addedTime(); case TR_SEED_DATE: return torrent->completedTime(); case TR_TRACKER: return torrent->currentTracker(); case TR_DLLIMIT: return torrent->downloadLimit(); case TR_UPLIMIT: return torrent->uploadLimit(); case TR_AMOUNT_DOWNLOADED: return torrent->totalDownload(); case TR_AMOUNT_UPLOADED: return torrent->totalUpload(); case TR_AMOUNT_DOWNLOADED_SESSION: return torrent->totalPayloadDownload(); case TR_AMOUNT_UPLOADED_SESSION: 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: return torrent->completedSize(); case TR_RATIO_LIMIT: return torrent->maxRatio(); case TR_SEEN_COMPLETE_DATE: return torrent->lastSeenComplete(); case TR_LAST_ACTIVITY: return torrent->timeSinceActivity(); 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()); } 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 {}; switch (role) { case Qt::ForegroundRole: return m_stateThemeColors.value(torrent->state(), getDefaultColorByState(torrent->state())); case Qt::DisplayRole: return displayValue(torrent, index.column()); case UnderlyingDataRole: 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: 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: 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: return QVariant(Qt::AlignRight | Qt::AlignVCenter); } break; default: break; } return {}; } bool TransferListModel::setData(const QModelIndex &index, const QVariant &value, int role) { if (!index.isValid() || (role != Qt::DisplayRole)) return false; BitTorrent::Torrent *const torrent = m_torrentList.value(index.row()); if (!torrent) return false; // Category and Name columns can be edited switch (index.column()) { case TR_NAME: torrent->setName(value.toString()); break; case TR_CATEGORY: torrent->setCategory(value.toString()); break; default: return false; } return true; } void TransferListModel::addTorrents(const QVector &torrents) { int row = m_torrentList.size(); beginInsertRows({}, row, (row + torrents.size())); 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 { if (!index.isValid()) return Qt::NoItemFlags; // Explicitly mark as editable return QAbstractListModel::flags(index) | Qt::ItemIsEditable; } BitTorrent::Torrent *TransferListModel::torrentHandle(const QModelIndex &index) const { if (!index.isValid()) return nullptr; return m_torrentList.value(index.row()); } void TransferListModel::handleTorrentAboutToBeRemoved(BitTorrent::Torrent *const torrent) { const int row = m_torrentMap.value(torrent, -1); Q_ASSERT(row >= 0); beginRemoveRows({}, row, row); m_torrentList.removeAt(row); m_torrentMap.remove(torrent); for (int &value : m_torrentMap) { if (value > row) --value; } endRemoveRows(); } void TransferListModel::handleTorrentStatusUpdated(BitTorrent::Torrent *const torrent) { const int row = m_torrentMap.value(torrent, -1); Q_ASSERT(row >= 0); emit dataChanged(index(row, 0), index(row, columnCount() - 1)); } void TransferListModel::handleTorrentsUpdated(const QVector &torrents) { const int columns = (columnCount() - 1); if (torrents.size() <= (m_torrentList.size() * 0.5)) { for (BitTorrent::Torrent *const torrent : torrents) { const int row = m_torrentMap.value(torrent, -1); Q_ASSERT(row >= 0); emit dataChanged(index(row, 0), index(row, columns)); } } else { // 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; if (pref->getHideZeroValues()) { if (pref->getHideZeroComboValues() == 1) hideZeroValuesMode = HideZeroValuesMode::Paused; else hideZeroValuesMode = HideZeroValuesMode::Always; } 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; } }