2012-02-21 23:54:24 +04:00
|
|
|
/*
|
2018-06-06 16:48:17 +03:00
|
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
|
|
* Copyright (C) 2006-2012 Christophe Dumez <chris@qbittorrent.org>
|
2012-02-21 23:54:24 +04: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.
|
|
|
|
*/
|
|
|
|
|
2018-06-14 14:46:50 +03:00
|
|
|
#include "torrentcontentmodel.h"
|
|
|
|
|
2019-01-11 11:05:57 +03:00
|
|
|
#include <algorithm>
|
|
|
|
|
2016-12-30 03:14:13 +03:00
|
|
|
#include <QFileIconProvider>
|
|
|
|
#include <QFileInfo>
|
2015-04-19 18:17:47 +03:00
|
|
|
#include <QIcon>
|
|
|
|
|
2017-07-19 13:30:01 +03:00
|
|
|
#if defined(Q_OS_WIN)
|
2016-12-30 03:14:13 +03:00
|
|
|
#include <Windows.h>
|
|
|
|
#include <Shellapi.h>
|
2021-03-19 10:28:40 +03:00
|
|
|
#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
|
2016-12-30 03:14:13 +03:00
|
|
|
#include <QtWin>
|
2021-03-19 10:28:40 +03:00
|
|
|
#endif
|
2017-05-02 12:38:04 +03:00
|
|
|
#else
|
|
|
|
#include <QMimeDatabase>
|
|
|
|
#include <QMimeType>
|
2016-12-30 03:14:13 +03:00
|
|
|
#endif
|
|
|
|
|
2019-09-05 15:11:33 +03:00
|
|
|
#if defined Q_OS_WIN || defined Q_OS_MACOS
|
2017-08-15 13:01:21 +03:00
|
|
|
#define QBT_PIXMAP_CACHE_FOR_FILE_ICONS
|
|
|
|
#include <QPixmapCache>
|
|
|
|
#endif
|
|
|
|
|
2021-12-09 13:05:49 +03:00
|
|
|
#include "base/bittorrent/abstractfilestorage.h"
|
2019-03-06 08:58:07 +03:00
|
|
|
#include "base/bittorrent/downloadpriority.h"
|
2018-11-18 21:40:37 +03:00
|
|
|
#include "base/global.h"
|
2022-02-08 06:03:48 +03:00
|
|
|
#include "base/path.h"
|
2015-09-25 11:10:05 +03:00
|
|
|
#include "base/utils/fs.h"
|
2012-09-03 01:11:46 +04:00
|
|
|
#include "torrentcontentmodelfile.h"
|
2018-06-06 16:48:17 +03:00
|
|
|
#include "torrentcontentmodelfolder.h"
|
|
|
|
#include "torrentcontentmodelitem.h"
|
2019-07-16 07:01:33 +03:00
|
|
|
#include "uithememanager.h"
|
2018-06-06 16:48:17 +03:00
|
|
|
|
2019-09-05 15:11:33 +03:00
|
|
|
#ifdef Q_OS_MACOS
|
2017-08-11 08:37:06 +03:00
|
|
|
#include "macutilities.h"
|
2017-07-19 13:30:01 +03:00
|
|
|
#endif
|
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
namespace
|
|
|
|
{
|
2018-06-06 16:48:17 +03:00
|
|
|
class UnifiedFileIconProvider : public QFileIconProvider
|
2016-01-08 17:19:04 +03:00
|
|
|
{
|
2016-12-30 03:14:13 +03:00
|
|
|
public:
|
2022-05-26 22:46:14 +03:00
|
|
|
UnifiedFileIconProvider()
|
2021-11-21 18:59:38 +03:00
|
|
|
: m_textPlainIcon {UIThemeManager::instance()->getIcon(u"help-about"_qs)}
|
2022-05-26 22:46:14 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-12-30 03:14:13 +03:00
|
|
|
using QFileIconProvider::icon;
|
2017-07-19 13:30:01 +03:00
|
|
|
|
2022-05-26 22:46:14 +03:00
|
|
|
QIcon icon(const QFileInfo &) const override
|
2016-12-30 03:14:13 +03:00
|
|
|
{
|
2022-05-26 22:46:14 +03:00
|
|
|
return m_textPlainIcon;
|
2016-12-30 03:14:13 +03:00
|
|
|
}
|
2022-05-26 22:46:14 +03:00
|
|
|
|
|
|
|
private:
|
|
|
|
QIcon m_textPlainIcon;
|
2016-12-30 03:14:13 +03:00
|
|
|
};
|
2017-08-15 13:01:21 +03:00
|
|
|
|
|
|
|
#ifdef QBT_PIXMAP_CACHE_FOR_FILE_ICONS
|
2018-06-06 16:48:17 +03:00
|
|
|
class CachingFileIconProvider : public UnifiedFileIconProvider
|
2016-12-30 03:14:13 +03:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
using QFileIconProvider::icon;
|
2017-07-19 13:30:01 +03:00
|
|
|
|
2020-04-19 05:42:11 +03:00
|
|
|
QIcon icon(const QFileInfo &info) const final
|
2017-08-15 13:01:21 +03:00
|
|
|
{
|
|
|
|
const QString ext = info.suffix();
|
2020-11-16 10:02:11 +03:00
|
|
|
if (!ext.isEmpty())
|
|
|
|
{
|
2017-08-15 13:01:21 +03:00
|
|
|
QPixmap cached;
|
2022-05-26 22:06:04 +03:00
|
|
|
if (QPixmapCache::find(ext, &cached))
|
|
|
|
return {cached};
|
2017-08-15 13:01:21 +03:00
|
|
|
|
|
|
|
const QPixmap pixmap = pixmapForExtension(ext);
|
2020-11-16 10:02:11 +03:00
|
|
|
if (!pixmap.isNull())
|
|
|
|
{
|
2017-08-15 13:01:21 +03:00
|
|
|
QPixmapCache::insert(ext, pixmap);
|
2019-02-14 20:16:42 +03:00
|
|
|
return {pixmap};
|
2017-08-15 13:01:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return UnifiedFileIconProvider::icon(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual QPixmap pixmapForExtension(const QString &ext) const = 0;
|
|
|
|
};
|
2018-06-06 16:48:17 +03:00
|
|
|
#endif // QBT_PIXMAP_CACHE_FOR_FILE_ICONS
|
2017-08-15 13:01:21 +03:00
|
|
|
|
|
|
|
#if defined(Q_OS_WIN)
|
|
|
|
// See QTBUG-25319 for explanation why this is required
|
2018-06-06 16:48:17 +03:00
|
|
|
class WinShellFileIconProvider final : public CachingFileIconProvider
|
2017-08-15 13:01:21 +03:00
|
|
|
{
|
|
|
|
QPixmap pixmapForExtension(const QString &ext) const override
|
2016-12-30 03:14:13 +03:00
|
|
|
{
|
2022-03-23 10:27:47 +03:00
|
|
|
const std::wstring extWStr = QString(u'.' + ext).toStdWString();
|
2022-05-26 22:06:04 +03:00
|
|
|
|
|
|
|
SHFILEINFOW sfi {};
|
|
|
|
const HRESULT hr = ::SHGetFileInfoW(extWStr.c_str(),
|
|
|
|
FILE_ATTRIBUTE_NORMAL, &sfi, sizeof(sfi), (SHGFI_ICON | SHGFI_USEFILEATTRIBUTES));
|
2016-12-30 03:14:13 +03:00
|
|
|
if (FAILED(hr))
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2016-12-30 03:14:13 +03:00
|
|
|
|
2021-03-19 10:28:40 +03:00
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
2022-05-26 22:06:04 +03:00
|
|
|
const auto iconPixmap = QPixmap::fromImage(QImage::fromHICON(sfi.hIcon));
|
2021-03-19 10:28:40 +03:00
|
|
|
#else
|
2022-05-26 22:06:04 +03:00
|
|
|
const QPixmap iconPixmap = QtWin::fromHICON(sfi.hIcon);
|
2021-03-19 10:28:40 +03:00
|
|
|
#endif
|
2016-12-30 03:14:13 +03:00
|
|
|
::DestroyIcon(sfi.hIcon);
|
2017-08-15 13:01:21 +03:00
|
|
|
return iconPixmap;
|
2016-12-30 03:14:13 +03:00
|
|
|
}
|
|
|
|
};
|
2019-09-05 15:11:33 +03:00
|
|
|
#elif defined(Q_OS_MACOS)
|
2017-07-19 13:30:01 +03:00
|
|
|
// There is a similar bug on macOS, to be reported to Qt
|
|
|
|
// https://github.com/qbittorrent/qBittorrent/pull/6156#issuecomment-316302615
|
2018-11-06 18:49:17 +03:00
|
|
|
class MacFileIconProvider final : public CachingFileIconProvider
|
2017-07-19 13:30:01 +03:00
|
|
|
{
|
2017-08-15 13:01:21 +03:00
|
|
|
QPixmap pixmapForExtension(const QString &ext) const override
|
2017-07-19 13:30:01 +03:00
|
|
|
{
|
2017-12-12 20:20:48 +03:00
|
|
|
return MacUtils::pixmapForExtension(ext, QSize(32, 32));
|
2017-07-19 13:30:01 +03:00
|
|
|
}
|
|
|
|
};
|
2017-05-02 12:38:04 +03:00
|
|
|
#else
|
|
|
|
/**
|
|
|
|
* @brief Tests whether QFileIconProvider actually works
|
|
|
|
*
|
|
|
|
* Some QPA plugins do not implement QPlatformTheme::fileIcon(), and
|
|
|
|
* QFileIconProvider::icon() returns empty icons as the result. Here we ask it for
|
|
|
|
* two icons for probably absent files and when both icons are null, we assume that
|
|
|
|
* the current QPA plugin does not implement QPlatformTheme::fileIcon().
|
|
|
|
*/
|
|
|
|
bool doesQFileIconProviderWork()
|
|
|
|
{
|
2022-05-06 08:00:36 +03:00
|
|
|
const Path PSEUDO_UNIQUE_FILE_NAME = Utils::Fs::tempPath() / Path(u"qBittorrent-test-QFileIconProvider-845eb448-7ad5-4cdb-b764-b3f322a266a9"_qs);
|
2017-05-02 12:38:04 +03:00
|
|
|
QFileIconProvider provider;
|
2022-05-06 08:00:36 +03:00
|
|
|
const QIcon testIcon1 = provider.icon(QFileInfo((PSEUDO_UNIQUE_FILE_NAME + u".pdf").data()));
|
|
|
|
const QIcon testIcon2 = provider.icon(QFileInfo((PSEUDO_UNIQUE_FILE_NAME + u".png").data()));
|
2017-05-02 12:38:04 +03:00
|
|
|
return (!testIcon1.isNull() || !testIcon2.isNull());
|
|
|
|
}
|
|
|
|
|
2022-05-26 22:06:04 +03:00
|
|
|
class MimeFileIconProvider final : public UnifiedFileIconProvider
|
2017-05-02 12:38:04 +03:00
|
|
|
{
|
|
|
|
using QFileIconProvider::icon;
|
2017-07-19 13:30:01 +03:00
|
|
|
|
2017-05-02 12:38:04 +03:00
|
|
|
QIcon icon(const QFileInfo &info) const override
|
|
|
|
{
|
2022-05-26 22:06:04 +03:00
|
|
|
const QMimeType mimeType = QMimeDatabase().mimeTypeForFile(info, QMimeDatabase::MatchExtension);
|
2017-05-02 12:38:04 +03:00
|
|
|
|
2022-05-26 22:06:04 +03:00
|
|
|
const auto mimeIcon = QIcon::fromTheme(mimeType.iconName());
|
|
|
|
if (!mimeIcon.isNull())
|
|
|
|
return mimeIcon;
|
|
|
|
|
|
|
|
const auto genericIcon = QIcon::fromTheme(mimeType.genericIconName());
|
|
|
|
if (!genericIcon.isNull())
|
|
|
|
return genericIcon;
|
2017-05-02 12:38:04 +03:00
|
|
|
|
|
|
|
return UnifiedFileIconProvider::icon(info);
|
|
|
|
}
|
|
|
|
};
|
2018-06-06 16:48:17 +03:00
|
|
|
#endif // Q_OS_WIN
|
2014-11-09 11:52:28 +03:00
|
|
|
}
|
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
TorrentContentModel::TorrentContentModel(QObject *parent)
|
|
|
|
: QAbstractItemModel(parent)
|
2022-02-27 07:48:11 +03:00
|
|
|
, m_rootItem(new TorrentContentModelFolder(QVector<QString>({ tr("Name"), tr("Total Size"), tr("Progress"), tr("Download Priority"), tr("Remaining"), tr("Availability") })))
|
2017-07-19 13:30:01 +03:00
|
|
|
#if defined(Q_OS_WIN)
|
2022-05-26 22:46:14 +03:00
|
|
|
, m_fileIconProvider {new WinShellFileIconProvider}
|
2019-09-05 15:11:33 +03:00
|
|
|
#elif defined(Q_OS_MACOS)
|
2022-05-26 22:46:14 +03:00
|
|
|
, m_fileIconProvider {new MacFileIconProvider}
|
2016-12-30 03:14:13 +03:00
|
|
|
#else
|
2022-05-26 22:46:14 +03:00
|
|
|
, m_fileIconProvider {doesQFileIconProviderWork() ? new QFileIconProvider : new MimeFileIconProvider}
|
2016-12-30 03:14:13 +03:00
|
|
|
#endif
|
2022-05-26 22:46:14 +03:00
|
|
|
{
|
2022-07-24 07:46:19 +03:00
|
|
|
m_fileIconProvider->setOptions(QFileIconProvider::DontUseCustomDirectoryIcons);
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
TorrentContentModel::~TorrentContentModel()
|
|
|
|
{
|
2016-12-30 03:14:13 +03:00
|
|
|
delete m_fileIconProvider;
|
2016-01-08 17:19:04 +03:00
|
|
|
delete m_rootItem;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2015-04-19 18:17:47 +03:00
|
|
|
void TorrentContentModel::updateFilesProgress(const QVector<qreal> &fp)
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
Q_ASSERT(m_filesIndex.size() == fp.size());
|
|
|
|
// XXX: Why is this necessary?
|
|
|
|
if (m_filesIndex.size() != fp.size()) return;
|
|
|
|
|
|
|
|
emit layoutAboutToBeChanged();
|
|
|
|
for (int i = 0; i < fp.size(); ++i)
|
|
|
|
m_filesIndex[i]->setProgress(fp[i]);
|
|
|
|
// Update folders progress in the tree
|
|
|
|
m_rootItem->recalculateProgress();
|
2016-03-22 18:45:54 +03:00
|
|
|
m_rootItem->recalculateAvailability();
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2019-03-06 08:58:07 +03:00
|
|
|
void TorrentContentModel::updateFilesPriorities(const QVector<BitTorrent::DownloadPriority> &fprio)
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2017-05-29 15:10:31 +03:00
|
|
|
Q_ASSERT(m_filesIndex.size() == fprio.size());
|
2016-01-08 17:19:04 +03:00
|
|
|
// XXX: Why is this necessary?
|
2017-05-29 15:10:31 +03:00
|
|
|
if (m_filesIndex.size() != fprio.size())
|
2016-01-08 17:19:04 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
emit layoutAboutToBeChanged();
|
|
|
|
for (int i = 0; i < fprio.size(); ++i)
|
2019-03-06 08:58:07 +03:00
|
|
|
m_filesIndex[i]->setPriority(static_cast<BitTorrent::DownloadPriority>(fprio[i]));
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2016-03-22 18:45:54 +03:00
|
|
|
void TorrentContentModel::updateFilesAvailability(const QVector<qreal> &fa)
|
|
|
|
{
|
|
|
|
Q_ASSERT(m_filesIndex.size() == fa.size());
|
|
|
|
// XXX: Why is this necessary?
|
|
|
|
if (m_filesIndex.size() != fa.size()) return;
|
|
|
|
|
|
|
|
emit layoutAboutToBeChanged();
|
2020-04-27 21:38:20 +03:00
|
|
|
for (int i = 0; i < m_filesIndex.size(); ++i)
|
2016-03-22 18:45:54 +03:00
|
|
|
m_filesIndex[i]->setAvailability(fa[i]);
|
|
|
|
// Update folders progress in the tree
|
|
|
|
m_rootItem->recalculateProgress();
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
|
2016-03-22 18:45:54 +03:00
|
|
|
}
|
|
|
|
|
2019-03-06 08:58:07 +03:00
|
|
|
QVector<BitTorrent::DownloadPriority> TorrentContentModel::getFilePriorities() const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2019-03-06 08:58:07 +03:00
|
|
|
QVector<BitTorrent::DownloadPriority> prio;
|
2016-01-08 17:19:04 +03:00
|
|
|
prio.reserve(m_filesIndex.size());
|
2018-11-27 23:15:04 +03:00
|
|
|
for (const TorrentContentModelFile *file : asConst(m_filesIndex))
|
2019-03-06 08:58:07 +03:00
|
|
|
prio.push_back(file->priority());
|
2016-01-08 17:19:04 +03:00
|
|
|
return prio;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool TorrentContentModel::allFiltered() const
|
|
|
|
{
|
2019-01-11 11:05:57 +03:00
|
|
|
return std::all_of(m_filesIndex.cbegin(), m_filesIndex.cend(), [](const TorrentContentModelFile *fileItem)
|
|
|
|
{
|
2019-03-06 08:58:07 +03:00
|
|
|
return (fileItem->priority() == BitTorrent::DownloadPriority::Ignored);
|
2019-01-11 11:05:57 +03:00
|
|
|
});
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
int TorrentContentModel::columnCount(const QModelIndex &parent) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (parent.isValid())
|
|
|
|
return static_cast<TorrentContentModelItem*>(parent.internalPointer())->columnCount();
|
2019-02-12 21:05:01 +03:00
|
|
|
|
|
|
|
return m_rootItem->columnCount();
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
bool TorrentContentModel::setData(const QModelIndex &index, const QVariant &value, int role)
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!index.isValid())
|
|
|
|
return false;
|
|
|
|
|
2020-11-16 10:02:11 +03:00
|
|
|
if ((index.column() == TorrentContentModelItem::COL_NAME) && (role == Qt::CheckStateRole))
|
|
|
|
{
|
2019-02-13 18:12:02 +03:00
|
|
|
auto *item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
2021-10-02 21:42:58 +03:00
|
|
|
qDebug("setData(%s, %d)", qUtf8Printable(item->name()), value.toInt());
|
2018-10-22 01:58:11 +03:00
|
|
|
|
2021-11-16 08:50:11 +03:00
|
|
|
BitTorrent::DownloadPriority prio = BitTorrent::DownloadPriority::Normal;
|
|
|
|
if (value.toInt() == Qt::PartiallyChecked)
|
|
|
|
prio = BitTorrent::DownloadPriority::Mixed;
|
|
|
|
else if (value.toInt() == Qt::Unchecked)
|
|
|
|
prio = BitTorrent::DownloadPriority::Ignored;
|
|
|
|
|
|
|
|
if (item->priority() != prio)
|
|
|
|
{
|
2018-10-22 01:58:11 +03:00
|
|
|
item->setPriority(prio);
|
2016-01-08 17:19:04 +03:00
|
|
|
// Update folders progress in the tree
|
|
|
|
m_rootItem->recalculateProgress();
|
2016-03-22 18:45:54 +03:00
|
|
|
m_rootItem->recalculateAvailability();
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(this->index(0, 0), this->index((rowCount() - 1), (columnCount() - 1)));
|
2016-01-08 17:19:04 +03:00
|
|
|
emit filteredFilesChanged();
|
|
|
|
}
|
|
|
|
return true;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2020-11-16 10:02:11 +03:00
|
|
|
if (role == Qt::EditRole)
|
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
Q_ASSERT(index.isValid());
|
2019-02-13 18:12:02 +03:00
|
|
|
auto *item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
2020-11-16 10:02:11 +03:00
|
|
|
switch (index.column())
|
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
case TorrentContentModelItem::COL_NAME:
|
|
|
|
item->setName(value.toString());
|
|
|
|
break;
|
|
|
|
case TorrentContentModelItem::COL_PRIO:
|
2022-05-18 07:47:48 +03:00
|
|
|
{
|
|
|
|
const BitTorrent::DownloadPriority previousPrio = item->priority();
|
|
|
|
const auto newPrio = static_cast<BitTorrent::DownloadPriority>(value.toInt());
|
|
|
|
item->setPriority(newPrio);
|
|
|
|
if ((newPrio != previousPrio) && ((newPrio == BitTorrent::DownloadPriority::Ignored)
|
|
|
|
|| (previousPrio == BitTorrent::DownloadPriority::Ignored)))
|
|
|
|
{
|
|
|
|
emit filteredFilesChanged();
|
|
|
|
}
|
|
|
|
}
|
2016-01-08 17:19:04 +03:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
emit dataChanged(index, index);
|
|
|
|
return true;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
2012-08-26 21:08:26 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
return false;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
TorrentContentModelItem::ItemType TorrentContentModel::itemType(const QModelIndex &index) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
return static_cast<const TorrentContentModelItem*>(index.internalPointer())->itemType();
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
int TorrentContentModel::getFileIndex(const QModelIndex &index)
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2019-02-13 18:12:02 +03:00
|
|
|
auto *item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
2016-01-15 11:32:16 +03:00
|
|
|
if (item->itemType() == TorrentContentModelItem::FileType)
|
|
|
|
return static_cast<TorrentContentModelFile*>(item)->fileIndex();
|
|
|
|
|
|
|
|
Q_ASSERT(item->itemType() == TorrentContentModelItem::FileType);
|
|
|
|
return -1;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2021-07-08 09:21:25 +03:00
|
|
|
QVariant TorrentContentModel::data(const QModelIndex &index, const int role) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!index.isValid())
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2019-02-13 18:12:02 +03:00
|
|
|
auto *item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
2016-12-25 01:18:37 +03:00
|
|
|
|
2020-11-16 10:02:11 +03:00
|
|
|
switch (role)
|
|
|
|
{
|
|
|
|
case Qt::DecorationRole:
|
2021-10-02 21:42:58 +03:00
|
|
|
{
|
2020-10-24 10:27:46 +03:00
|
|
|
if (index.column() != TorrentContentModelItem::COL_NAME)
|
|
|
|
return {};
|
2019-02-12 21:05:01 +03:00
|
|
|
|
2020-10-24 10:27:46 +03:00
|
|
|
if (item->itemType() == TorrentContentModelItem::FolderType)
|
|
|
|
return m_fileIconProvider->icon(QFileIconProvider::Folder);
|
|
|
|
return m_fileIconProvider->icon(QFileInfo(item->name()));
|
|
|
|
}
|
2020-11-16 10:02:11 +03:00
|
|
|
case Qt::CheckStateRole:
|
2021-10-02 21:42:58 +03:00
|
|
|
{
|
2020-10-24 10:27:46 +03:00
|
|
|
if (index.column() != TorrentContentModelItem::COL_NAME)
|
|
|
|
return {};
|
2016-12-25 01:18:37 +03:00
|
|
|
|
2020-10-24 10:27:46 +03:00
|
|
|
if (item->priority() == BitTorrent::DownloadPriority::Ignored)
|
|
|
|
return Qt::Unchecked;
|
|
|
|
if (item->priority() == BitTorrent::DownloadPriority::Mixed)
|
|
|
|
return Qt::PartiallyChecked;
|
|
|
|
return Qt::Checked;
|
|
|
|
}
|
2020-10-28 08:09:05 +03:00
|
|
|
case Qt::TextAlignmentRole:
|
|
|
|
if ((index.column() == TorrentContentModelItem::COL_SIZE)
|
|
|
|
|| (index.column() == TorrentContentModelItem::COL_REMAINING))
|
|
|
|
return QVariant {Qt::AlignRight | Qt::AlignVCenter};
|
|
|
|
return {};
|
|
|
|
|
2020-10-24 10:27:46 +03:00
|
|
|
case Qt::DisplayRole:
|
2021-07-08 09:21:25 +03:00
|
|
|
case Qt::ToolTipRole:
|
2020-10-24 10:27:46 +03:00
|
|
|
return item->displayData(index.column());
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2020-10-24 10:27:46 +03:00
|
|
|
case Roles::UnderlyingDataRole:
|
|
|
|
return item->underlyingData(index.column());
|
2016-12-25 01:18:37 +03:00
|
|
|
|
2020-10-24 10:27:46 +03:00
|
|
|
default:
|
|
|
|
return {};
|
|
|
|
}
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
Qt::ItemFlags TorrentContentModel::flags(const QModelIndex &index) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!index.isValid())
|
2018-09-07 14:12:38 +03:00
|
|
|
return Qt::NoItemFlags;
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2021-01-05 16:38:00 +03:00
|
|
|
Qt::ItemFlags flags {Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable};
|
2016-01-08 17:19:04 +03:00
|
|
|
if (itemType(index) == TorrentContentModelItem::FolderType)
|
2021-02-19 17:51:53 +03:00
|
|
|
flags |= Qt::ItemIsAutoTristate;
|
2021-01-05 16:38:00 +03:00
|
|
|
if (index.column() == TorrentContentModelItem::COL_PRIO)
|
|
|
|
flags |= Qt::ItemIsEditable;
|
2012-08-26 21:08:26 +04:00
|
|
|
|
2021-01-05 16:38:00 +03:00
|
|
|
return flags;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
QVariant TorrentContentModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
2020-10-28 08:09:05 +03:00
|
|
|
if (orientation != Qt::Horizontal)
|
|
|
|
return {};
|
|
|
|
|
2020-11-16 10:02:11 +03:00
|
|
|
switch (role)
|
|
|
|
{
|
2020-10-28 08:09:05 +03:00
|
|
|
case Qt::DisplayRole:
|
2020-10-24 10:27:46 +03:00
|
|
|
return m_rootItem->displayData(section);
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2020-10-28 08:09:05 +03:00
|
|
|
case Qt::TextAlignmentRole:
|
|
|
|
if ((section == TorrentContentModelItem::COL_SIZE)
|
|
|
|
|| (section == TorrentContentModelItem::COL_REMAINING))
|
|
|
|
return QVariant {Qt::AlignRight | Qt::AlignVCenter};
|
|
|
|
return {};
|
|
|
|
|
|
|
|
default:
|
|
|
|
return {};
|
|
|
|
}
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
QModelIndex TorrentContentModel::index(int row, int column, const QModelIndex &parent) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (parent.isValid() && (parent.column() != 0))
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
if (column >= TorrentContentModelItem::NB_COL)
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2022-06-23 07:28:00 +03:00
|
|
|
TorrentContentModelFolder *parentItem = nullptr;
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!parent.isValid())
|
|
|
|
parentItem = m_rootItem;
|
|
|
|
else
|
|
|
|
parentItem = static_cast<TorrentContentModelFolder*>(parent.internalPointer());
|
|
|
|
Q_ASSERT(parentItem);
|
2012-08-26 21:08:26 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
if (row >= parentItem->childCount())
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
TorrentContentModelItem *childItem = parentItem->child(row);
|
2016-01-08 17:19:04 +03:00
|
|
|
if (childItem)
|
|
|
|
return createIndex(row, column, childItem);
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
QModelIndex TorrentContentModel::parent(const QModelIndex &index) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!index.isValid())
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2019-02-13 18:12:02 +03:00
|
|
|
auto *childItem = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!childItem)
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
TorrentContentModelItem *parentItem = childItem->parent();
|
|
|
|
if (parentItem == m_rootItem)
|
2019-02-14 20:16:42 +03:00
|
|
|
return {};
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
return createIndex(parentItem->row(), 0, parentItem);
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2018-06-06 16:48:17 +03:00
|
|
|
int TorrentContentModel::rowCount(const QModelIndex &parent) const
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
if (parent.column() > 0)
|
|
|
|
return 0;
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2022-06-23 07:28:00 +03:00
|
|
|
TorrentContentModelFolder *parentItem = nullptr;
|
2016-01-08 17:19:04 +03:00
|
|
|
if (!parent.isValid())
|
|
|
|
parentItem = m_rootItem;
|
|
|
|
else
|
|
|
|
parentItem = dynamic_cast<TorrentContentModelFolder*>(static_cast<TorrentContentModelItem*>(parent.internalPointer()));
|
2012-02-21 23:54:24 +04:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
return parentItem ? parentItem->childCount() : 0;
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentContentModel::clear()
|
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
qDebug("clear called");
|
|
|
|
beginResetModel();
|
|
|
|
m_filesIndex.clear();
|
|
|
|
m_rootItem->deleteAllChildren();
|
|
|
|
endResetModel();
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
2021-12-09 13:05:49 +03:00
|
|
|
void TorrentContentModel::setupModelData(const BitTorrent::AbstractFileStorage &info)
|
2012-02-21 23:54:24 +04:00
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
qDebug("setup model data called");
|
2017-02-08 08:11:36 +03:00
|
|
|
const int filesCount = info.filesCount();
|
|
|
|
if (filesCount <= 0)
|
2016-01-08 17:19:04 +03:00
|
|
|
return;
|
|
|
|
|
|
|
|
emit layoutAboutToBeChanged();
|
|
|
|
// Initialize files_index array
|
2017-02-08 08:11:36 +03:00
|
|
|
qDebug("Torrent contains %d files", filesCount);
|
|
|
|
m_filesIndex.reserve(filesCount);
|
2016-01-08 17:19:04 +03:00
|
|
|
|
2022-01-24 05:55:06 +03:00
|
|
|
QHash<TorrentContentModelFolder *, QHash<QString, TorrentContentModelFolder *>> folderMap;
|
2022-06-15 20:14:53 +03:00
|
|
|
QVector<QString> lastParentPath;
|
|
|
|
TorrentContentModelFolder *lastParent = m_rootItem;
|
2016-01-08 17:19:04 +03:00
|
|
|
// Iterate over files
|
2020-11-16 10:02:11 +03:00
|
|
|
for (int i = 0; i < filesCount; ++i)
|
|
|
|
{
|
2022-02-08 06:03:48 +03:00
|
|
|
const QString path = info.filePath(i).data();
|
2019-08-06 18:07:57 +03:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
// Iterate of parts of the path to create necessary folders
|
2021-03-01 09:40:30 +03:00
|
|
|
QList<QStringView> pathFolders = QStringView(path).split(u'/', Qt::SkipEmptyParts);
|
2022-06-15 20:14:53 +03:00
|
|
|
const QString fileName = pathFolders.takeLast().toString();
|
2019-08-06 18:07:57 +03:00
|
|
|
|
2022-06-15 20:14:53 +03:00
|
|
|
if (!std::equal(lastParentPath.begin(), lastParentPath.end()
|
|
|
|
, pathFolders.begin(), pathFolders.end()))
|
2020-11-16 10:02:11 +03:00
|
|
|
{
|
2022-06-15 20:14:53 +03:00
|
|
|
lastParentPath.clear();
|
|
|
|
lastParentPath.reserve(pathFolders.size());
|
|
|
|
|
|
|
|
// rebuild the path from the root
|
|
|
|
lastParent = m_rootItem;
|
|
|
|
for (const QStringView pathPart : asConst(pathFolders))
|
2020-11-16 10:02:11 +03:00
|
|
|
{
|
2022-06-15 20:14:53 +03:00
|
|
|
const QString folderName = pathPart.toString();
|
|
|
|
lastParentPath.push_back(folderName);
|
2022-01-24 05:55:06 +03:00
|
|
|
|
2022-06-15 20:14:53 +03:00
|
|
|
TorrentContentModelFolder *&newParent = folderMap[lastParent][folderName];
|
|
|
|
if (!newParent)
|
|
|
|
{
|
|
|
|
newParent = new TorrentContentModelFolder(folderName, lastParent);
|
|
|
|
lastParent->appendChild(newParent);
|
|
|
|
}
|
|
|
|
|
|
|
|
lastParent = newParent;
|
|
|
|
}
|
2016-01-08 17:19:04 +03:00
|
|
|
}
|
2022-06-15 20:14:53 +03:00
|
|
|
|
2016-01-08 17:19:04 +03:00
|
|
|
// Actually create the file
|
2021-10-02 21:42:58 +03:00
|
|
|
TorrentContentModelFile *fileItem = new TorrentContentModelFile(
|
2022-06-15 20:14:53 +03:00
|
|
|
fileName, info.fileSize(i), lastParent, i);
|
|
|
|
lastParent->appendChild(fileItem);
|
2016-01-08 17:19:04 +03:00
|
|
|
m_filesIndex.push_back(fileItem);
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
2016-01-08 17:19:04 +03:00
|
|
|
emit layoutChanged();
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentContentModel::selectAll()
|
|
|
|
{
|
2020-11-16 10:02:11 +03:00
|
|
|
for (int i = 0; i < m_rootItem->childCount(); ++i)
|
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
TorrentContentModelItem* child = m_rootItem->child(i);
|
2019-03-06 08:58:07 +03:00
|
|
|
if (child->priority() == BitTorrent::DownloadPriority::Ignored)
|
|
|
|
child->setPriority(BitTorrent::DownloadPriority::Normal);
|
2016-01-08 17:19:04 +03:00
|
|
|
}
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentContentModel::selectNone()
|
|
|
|
{
|
2016-01-08 17:19:04 +03:00
|
|
|
for (int i = 0; i < m_rootItem->childCount(); ++i)
|
2019-03-06 08:58:07 +03:00
|
|
|
m_rootItem->child(i)->setPriority(BitTorrent::DownloadPriority::Ignored);
|
2019-08-17 10:53:51 +03:00
|
|
|
emit dataChanged(index(0, 0), index((rowCount() - 1), (columnCount() - 1)));
|
2012-02-21 23:54:24 +04:00
|
|
|
}
|