mirror of
https://github.com/qbittorrent/qBittorrent.git
synced 2025-01-09 08:37:27 +03:00
ff9a281b72
Change project directory structure according to application structure. Change 'nox' configuration option to something more meaningful 'nogui'. Rename 'Icons' folder to 'icons' (similar to other folders). Partially add 'nowebui' option support. Remove QConf project file.
337 lines
10 KiB
C++
337 lines
10 KiB
C++
/*
|
|
* Bittorrent Client using Qt4 and libtorrent.
|
|
* Copyright (C) 2006-2012 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.
|
|
*
|
|
* Contact : chris@qbittorrent.org
|
|
*/
|
|
|
|
#include "iconprovider.h"
|
|
#include "misc.h"
|
|
#include "fs_utils.h"
|
|
#include "torrentcontentmodel.h"
|
|
#include "torrentcontentmodelitem.h"
|
|
#include "torrentcontentmodelfolder.h"
|
|
#include "torrentcontentmodelfile.h"
|
|
#include <QDir>
|
|
|
|
namespace {
|
|
QIcon get_directory_icon() {
|
|
static QIcon cached = IconProvider::instance()->getIcon("inode-directory");
|
|
return cached;
|
|
}
|
|
|
|
QIcon get_file_icon() {
|
|
static QIcon cached = IconProvider::instance()->getIcon("text-plain");
|
|
return cached;
|
|
}
|
|
}
|
|
|
|
TorrentContentModel::TorrentContentModel(QObject *parent):
|
|
QAbstractItemModel(parent),
|
|
m_rootItem(new TorrentContentModelFolder(QList<QVariant>() << tr("Name") << tr("Size")
|
|
<< tr("Progress") << tr("Priority")))
|
|
{
|
|
}
|
|
|
|
TorrentContentModel::~TorrentContentModel()
|
|
{
|
|
delete m_rootItem;
|
|
}
|
|
|
|
void TorrentContentModel::updateFilesProgress(const std::vector<libtorrent::size_type>& fp)
|
|
{
|
|
Q_ASSERT(m_filesIndex.size() == (int)fp.size());
|
|
// XXX: Why is this necessary?
|
|
if (m_filesIndex.size() != (int)fp.size())
|
|
return;
|
|
|
|
emit layoutAboutToBeChanged();
|
|
for (uint i = 0; i < fp.size(); ++i) {
|
|
m_filesIndex[i]->setProgress(fp[i]);
|
|
}
|
|
// Update folders progress in the tree
|
|
m_rootItem->recalculateProgress();
|
|
emit dataChanged(index(0,0), index(rowCount(), columnCount()));
|
|
}
|
|
|
|
void TorrentContentModel::updateFilesPriorities(const std::vector<int>& fprio)
|
|
{
|
|
Q_ASSERT(m_filesIndex.size() == (int)fprio.size());
|
|
// XXX: Why is this necessary?
|
|
if (m_filesIndex.size() != (int)fprio.size())
|
|
return;
|
|
|
|
emit layoutAboutToBeChanged();
|
|
for (uint i = 0; i < fprio.size(); ++i) {
|
|
m_filesIndex[i]->setPriority(fprio[i]);
|
|
}
|
|
emit dataChanged(index(0, 0), index(rowCount(), columnCount()));
|
|
}
|
|
|
|
std::vector<int> TorrentContentModel::getFilesPriorities() const
|
|
{
|
|
std::vector<int> prio;
|
|
prio.reserve(m_filesIndex.size());
|
|
foreach (const TorrentContentModelFile* file, m_filesIndex) {
|
|
prio.push_back(file->priority());
|
|
}
|
|
return prio;
|
|
}
|
|
|
|
bool TorrentContentModel::allFiltered() const
|
|
{
|
|
foreach (const TorrentContentModelFile* fileItem, m_filesIndex) {
|
|
if (fileItem->priority() != prio::IGNORED)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
int TorrentContentModel::columnCount(const QModelIndex& parent) const
|
|
{
|
|
if (parent.isValid())
|
|
return static_cast<TorrentContentModelItem*>(parent.internalPointer())->columnCount();
|
|
else
|
|
return m_rootItem->columnCount();
|
|
}
|
|
|
|
bool TorrentContentModel::setData(const QModelIndex& index, const QVariant& value, int role)
|
|
{
|
|
if (!index.isValid())
|
|
return false;
|
|
|
|
if (index.column() == 0 && role == Qt::CheckStateRole) {
|
|
TorrentContentModelItem *item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
|
qDebug("setData(%s, %d", qPrintable(item->name()), value.toInt());
|
|
if (item->priority() != value.toInt()) {
|
|
if (value.toInt() == Qt::PartiallyChecked)
|
|
item->setPriority(prio::MIXED);
|
|
else if (value.toInt() == Qt::Unchecked)
|
|
item->setPriority(prio::IGNORED);
|
|
else
|
|
item->setPriority(prio::NORMAL);
|
|
// Update folders progress in the tree
|
|
m_rootItem->recalculateProgress();
|
|
emit dataChanged(this->index(0,0), this->index(rowCount()-1, columnCount()-1));
|
|
emit filteredFilesChanged();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
if (role == Qt::EditRole) {
|
|
Q_ASSERT(index.isValid());
|
|
TorrentContentModelItem* item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
|
switch(index.column()) {
|
|
case TorrentContentModelItem::COL_NAME:
|
|
item->setName(value.toString());
|
|
break;
|
|
case TorrentContentModelItem::COL_PRIO:
|
|
item->setPriority(value.toInt());
|
|
break;
|
|
default:
|
|
return false;
|
|
}
|
|
emit dataChanged(index, index);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
TorrentContentModelItem::ItemType TorrentContentModel::itemType(const QModelIndex& index) const
|
|
{
|
|
return static_cast<const TorrentContentModelItem*>(index.internalPointer())->itemType();
|
|
}
|
|
|
|
int TorrentContentModel::getFileIndex(const QModelIndex& index)
|
|
{
|
|
TorrentContentModelFile* item = dynamic_cast<TorrentContentModelFile*>(static_cast<TorrentContentModelItem*>(index.internalPointer()));
|
|
Q_ASSERT(item);
|
|
return item->fileIndex();
|
|
}
|
|
|
|
QVariant TorrentContentModel::data(const QModelIndex& index, int role) const
|
|
{
|
|
if (!index.isValid())
|
|
return QVariant();
|
|
|
|
TorrentContentModelItem* item = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
|
if (index.column() == 0 && role == Qt::DecorationRole) {
|
|
if (item->itemType() == TorrentContentModelItem::FolderType)
|
|
return get_directory_icon();
|
|
else
|
|
return get_file_icon();
|
|
}
|
|
if (index.column() == 0 && role == Qt::CheckStateRole) {
|
|
if (item->data(TorrentContentModelItem::COL_PRIO).toInt() == prio::IGNORED)
|
|
return Qt::Unchecked;
|
|
if (item->data(TorrentContentModelItem::COL_PRIO).toInt() == prio::MIXED)
|
|
return Qt::PartiallyChecked;
|
|
return Qt::Checked;
|
|
}
|
|
if (role != Qt::DisplayRole)
|
|
return QVariant();
|
|
|
|
return item->data(index.column());
|
|
}
|
|
|
|
Qt::ItemFlags TorrentContentModel::flags(const QModelIndex& index) const
|
|
{
|
|
if (!index.isValid())
|
|
return 0;
|
|
|
|
if (itemType(index) == TorrentContentModelItem::FolderType)
|
|
return Qt::ItemIsEditable | Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable | Qt::ItemIsTristate;
|
|
|
|
return Qt::ItemIsEditable | Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsUserCheckable;
|
|
}
|
|
|
|
QVariant TorrentContentModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
{
|
|
if (orientation == Qt::Horizontal && role == Qt::DisplayRole)
|
|
return m_rootItem->data(section);
|
|
|
|
return QVariant();
|
|
}
|
|
|
|
QModelIndex TorrentContentModel::index(int row, int column, const QModelIndex& parent) const
|
|
{
|
|
if (parent.isValid() && parent.column() != 0)
|
|
return QModelIndex();
|
|
|
|
if (column >= TorrentContentModelItem::NB_COL)
|
|
return QModelIndex();
|
|
|
|
TorrentContentModelFolder* parentItem;
|
|
if (!parent.isValid())
|
|
parentItem = m_rootItem;
|
|
else
|
|
parentItem = static_cast<TorrentContentModelFolder*>(parent.internalPointer());
|
|
Q_ASSERT(parentItem);
|
|
|
|
if (row >= parentItem->childCount())
|
|
return QModelIndex();
|
|
|
|
TorrentContentModelItem* childItem = parentItem->child(row);
|
|
if (childItem)
|
|
return createIndex(row, column, childItem);
|
|
return QModelIndex();
|
|
}
|
|
|
|
QModelIndex TorrentContentModel::parent(const QModelIndex& index) const
|
|
{
|
|
if (!index.isValid())
|
|
return QModelIndex();
|
|
|
|
TorrentContentModelItem* childItem = static_cast<TorrentContentModelItem*>(index.internalPointer());
|
|
if (!childItem)
|
|
return QModelIndex();
|
|
|
|
TorrentContentModelItem *parentItem = childItem->parent();
|
|
if (parentItem == m_rootItem)
|
|
return QModelIndex();
|
|
|
|
return createIndex(parentItem->row(), 0, parentItem);
|
|
}
|
|
|
|
int TorrentContentModel::rowCount(const QModelIndex& parent) const
|
|
{
|
|
if (parent.column() > 0)
|
|
return 0;
|
|
|
|
TorrentContentModelFolder* parentItem;
|
|
if (!parent.isValid())
|
|
parentItem = m_rootItem;
|
|
else
|
|
parentItem = dynamic_cast<TorrentContentModelFolder*>(static_cast<TorrentContentModelItem*>(parent.internalPointer()));
|
|
|
|
return parentItem ? parentItem->childCount() : 0;
|
|
}
|
|
|
|
void TorrentContentModel::clear()
|
|
{
|
|
qDebug("clear called");
|
|
beginResetModel();
|
|
m_filesIndex.clear();
|
|
m_rootItem->deleteAllChildren();
|
|
endResetModel();
|
|
}
|
|
|
|
void TorrentContentModel::setupModelData(const libtorrent::torrent_info& t)
|
|
{
|
|
qDebug("setup model data called");
|
|
if (t.num_files() == 0)
|
|
return;
|
|
|
|
emit layoutAboutToBeChanged();
|
|
// Initialize files_index array
|
|
qDebug("Torrent contains %d files", t.num_files());
|
|
m_filesIndex.reserve(t.num_files());
|
|
|
|
TorrentContentModelFolder* current_parent;
|
|
// Iterate over files
|
|
for (int i = 0; i < t.num_files(); ++i) {
|
|
const libtorrent::file_entry& fentry = t.file_at(i);
|
|
current_parent = m_rootItem;
|
|
QString path = fsutils::fromNativePath(misc::toQStringU(fentry.path));
|
|
// Iterate of parts of the path to create necessary folders
|
|
QStringList pathFolders = path.split("/", QString::SkipEmptyParts);
|
|
pathFolders.removeLast();
|
|
foreach (const QString& pathPart, pathFolders) {
|
|
if (pathPart == ".unwanted")
|
|
continue;
|
|
TorrentContentModelFolder* new_parent = current_parent->childFolderWithName(pathPart);
|
|
if (!new_parent) {
|
|
new_parent = new TorrentContentModelFolder(pathPart, current_parent);
|
|
current_parent->appendChild(new_parent);
|
|
}
|
|
current_parent = new_parent;
|
|
}
|
|
// Actually create the file
|
|
TorrentContentModelFile* fileItem = new TorrentContentModelFile(fentry, current_parent, i);
|
|
current_parent->appendChild(fileItem);
|
|
m_filesIndex.push_back(fileItem);
|
|
}
|
|
emit layoutChanged();
|
|
}
|
|
|
|
void TorrentContentModel::selectAll()
|
|
{
|
|
for (int i=0; i<m_rootItem->childCount(); ++i) {
|
|
TorrentContentModelItem* child = m_rootItem->child(i);
|
|
if (child->priority() == prio::IGNORED)
|
|
child->setPriority(prio::NORMAL);
|
|
}
|
|
emit dataChanged(index(0, 0), index(rowCount(), columnCount()));
|
|
}
|
|
|
|
void TorrentContentModel::selectNone()
|
|
{
|
|
for (int i=0; i<m_rootItem->childCount(); ++i) {
|
|
m_rootItem->child(i)->setPriority(prio::IGNORED);
|
|
}
|
|
emit dataChanged(index(0, 0), index(rowCount(), columnCount()));
|
|
}
|