1
0
Fork 0
mirror of https://github.com/transmission/transmission synced 2025-01-02 21:16:04 +00:00
transmission/qt/FileTreeModel.cc
dubhater 1e850cd874
Qt: Fix assert with one-letter root folder names
An assertion failure happens shortly after opening the
"Torrent properties" dialog if the torrent's root folder has
a one-letter name, e. g.
A/
A/Some Linux distro.iso
A/Some other Linux distro.iso

Off by one errors are great, eh?

This fixes https://trac.transmissionbt.com/ticket/6141
2018-01-30 23:44:12 +02:00

627 lines
14 KiB
C++

/*
* This file Copyright (C) 2009-2015 Mnemosyne LLC
*
* It may be used under the GNU GPL versions 2 or 3
* or any future license endorsed by Mnemosyne LLC.
*
*/
#include <cassert>
#include <libtransmission/transmission.h> // priorities
#include "FileTreeItem.h"
#include "FileTreeModel.h"
namespace
{
class PathIteratorBase
{
protected:
PathIteratorBase(QString const& path, int slashIndex) :
myPath(path),
mySlashIndex(slashIndex),
myToken()
{
myToken.reserve(path.size() / 2);
}
protected:
QString const& myPath;
int mySlashIndex;
QString myToken;
static QChar const SlashChar;
};
QChar const PathIteratorBase::SlashChar = QLatin1Char('/');
class ForwardPathIterator : public PathIteratorBase
{
public:
ForwardPathIterator(QString const& path) :
PathIteratorBase(path, path.size() - 1)
{
}
bool hasNext() const
{
return mySlashIndex > -1;
}
QString const& next()
{
int newSlashIndex = myPath.lastIndexOf(SlashChar, mySlashIndex);
myToken.truncate(0);
myToken += myPath.midRef(newSlashIndex + 1, mySlashIndex - newSlashIndex);
mySlashIndex = newSlashIndex - 1;
return myToken;
}
};
class BackwardPathIterator : public PathIteratorBase
{
public:
BackwardPathIterator(QString const& path) :
PathIteratorBase(path, 0)
{
}
bool hasNext() const
{
return mySlashIndex < myPath.size();
}
QString const& next()
{
int newSlashIndex = myPath.indexOf(SlashChar, mySlashIndex);
if (newSlashIndex == -1)
{
newSlashIndex = myPath.size();
}
myToken.truncate(0);
myToken += myPath.midRef(mySlashIndex, newSlashIndex - mySlashIndex);
mySlashIndex = newSlashIndex + 1;
return myToken;
}
};
} // namespace
FileTreeModel::FileTreeModel(QObject* parent, bool isEditable) :
QAbstractItemModel(parent),
myIsEditable(isEditable),
myRootItem(new FileTreeItem),
myIndexCache()
{
}
FileTreeModel::~FileTreeModel()
{
clear();
delete myRootItem;
}
void FileTreeModel::setEditable(bool editable)
{
myIsEditable = editable;
}
FileTreeItem* FileTreeModel::itemFromIndex(QModelIndex const& index) const
{
if (!index.isValid())
{
return nullptr;
}
assert(index.model() == this);
return static_cast<FileTreeItem*>(index.internalPointer());
}
QModelIndexList FileTreeModel::getOrphanIndices(QModelIndexList const& indices) const
{
QModelIndexList orphanIndices = indices;
qSort(orphanIndices);
for (QMutableListIterator<QModelIndex> it(orphanIndices); it.hasNext();)
{
QModelIndex walk = it.next();
for (;;)
{
walk = parent(walk, walk.column());
if (!walk.isValid())
{
break;
}
if (qBinaryFind(orphanIndices, walk) != orphanIndices.end())
{
it.remove();
break;
}
}
}
return orphanIndices;
}
QVariant FileTreeModel::data(QModelIndex const& index, int role) const
{
QVariant value;
if (index.isValid())
{
value = itemFromIndex(index)->data(index.column(), role);
}
return value;
}
Qt::ItemFlags FileTreeModel::flags(QModelIndex const& index) const
{
int i(Qt::ItemIsSelectable | Qt::ItemIsEnabled);
if (myIsEditable && index.column() == COL_NAME)
{
i |= Qt::ItemIsEditable;
}
if (index.column() == COL_WANTED)
{
i |= Qt::ItemIsUserCheckable | Qt::ItemIsTristate;
}
return Qt::ItemFlags(i);
}
bool FileTreeModel::setData(QModelIndex const& index, QVariant const& newname, int role)
{
if (role == Qt::EditRole)
{
FileTreeItem* item = itemFromIndex(index);
emit pathEdited(item->path(), newname.toString());
}
return false; // don't update the view until the session confirms the change
}
QVariant FileTreeModel::headerData(int column, Qt::Orientation orientation, int role) const
{
QVariant data;
if (orientation == Qt::Horizontal && role == Qt::DisplayRole)
{
switch (column)
{
case COL_NAME:
data.setValue(tr("File"));
break;
case COL_SIZE:
data.setValue(tr("Size"));
break;
case COL_PROGRESS:
data.setValue(tr("Progress"));
break;
case COL_WANTED:
data.setValue(tr("Download"));
break;
case COL_PRIORITY:
data.setValue(tr("Priority"));
break;
default:
break;
}
}
return data;
}
QModelIndex FileTreeModel::index(int row, int column, QModelIndex const& parent) const
{
QModelIndex i;
if (hasIndex(row, column, parent))
{
FileTreeItem* parentItem;
if (!parent.isValid())
{
parentItem = myRootItem;
}
else
{
parentItem = itemFromIndex(parent);
}
FileTreeItem* childItem = parentItem->child(row);
if (childItem != nullptr)
{
i = createIndex(row, column, childItem);
}
}
return i;
}
QModelIndex FileTreeModel::parent(QModelIndex const& child) const
{
return parent(child, 0); // QAbstractItemModel::parent() wants col 0
}
QModelIndex FileTreeModel::parent(QModelIndex const& child, int column) const
{
QModelIndex parent;
if (child.isValid())
{
parent = indexOf(itemFromIndex(child)->parent(), column);
}
return parent;
}
int FileTreeModel::rowCount(QModelIndex const& parent) const
{
FileTreeItem* parentItem;
if (parent.isValid())
{
parentItem = itemFromIndex(parent);
}
else
{
parentItem = myRootItem;
}
return parentItem->childCount();
}
int FileTreeModel::columnCount(QModelIndex const& parent) const
{
Q_UNUSED(parent);
return NUM_COLUMNS;
}
QModelIndex FileTreeModel::indexOf(FileTreeItem* item, int column) const
{
if (item == nullptr || item == myRootItem)
{
return QModelIndex();
}
return createIndex(item->row(), column, item);
}
void FileTreeModel::clearSubtree(QModelIndex const& top)
{
size_t i = rowCount(top);
while (i > 0)
{
clearSubtree(index(--i, 0, top));
}
FileTreeItem* const item = itemFromIndex(top);
if (item == nullptr)
{
return;
}
if (item->fileIndex() != -1)
{
myIndexCache.remove(item->fileIndex());
}
delete item;
}
void FileTreeModel::clear()
{
beginResetModel();
clearSubtree(QModelIndex());
endResetModel();
assert(myIndexCache.isEmpty());
}
FileTreeItem* FileTreeModel::findItemForFileIndex(int fileIndex) const
{
return myIndexCache.value(fileIndex, nullptr);
}
void FileTreeModel::addFile(int fileIndex, QString const& filename, bool wanted, int priority, uint64_t totalSize,
uint64_t have, bool updateFields)
{
FileTreeItem* item;
item = findItemForFileIndex(fileIndex);
if (item != nullptr) // this file is already in the tree, we've added this
{
QModelIndex indexWithChangedParents;
ForwardPathIterator filenameIt(filename);
while (filenameIt.hasNext())
{
QString const& token = filenameIt.next();
std::pair<int, int> const changed = item->update(token, wanted, priority, have, updateFields);
if (changed.first >= 0)
{
emit dataChanged(indexOf(item, changed.first), indexOf(item, changed.second));
if (!indexWithChangedParents.isValid() && changed.first <= COL_PRIORITY && changed.second >= COL_SIZE)
{
indexWithChangedParents = indexOf(item, 0);
}
}
item = item->parent();
}
assert(item == myRootItem);
if (indexWithChangedParents.isValid())
{
emitParentsChanged(indexWithChangedParents, COL_SIZE, COL_PRIORITY);
}
}
else // we haven't build the FileTreeItems for these tokens yet
{
bool added = false;
item = myRootItem;
BackwardPathIterator filenameIt(filename);
while (filenameIt.hasNext())
{
QString const& token = filenameIt.next();
FileTreeItem* child(item->child(token));
if (child == nullptr)
{
added = true;
QModelIndex parentIndex(indexOf(item, 0));
int const n(item->childCount());
beginInsertRows(parentIndex, n, n);
if (!filenameIt.hasNext())
{
child = new FileTreeItem(token, fileIndex, totalSize);
}
else
{
child = new FileTreeItem(token);
}
item->appendChild(child);
endInsertRows();
}
item = child;
}
if (item != myRootItem)
{
assert(item->fileIndex() == fileIndex);
assert(item->totalSize() == totalSize);
myIndexCache[fileIndex] = item;
std::pair<int, int> const changed = item->update(item->name(), wanted, priority, have, added || updateFields);
if (changed.first >= 0)
{
emit dataChanged(indexOf(item, changed.first), indexOf(item, changed.second));
}
}
}
}
void FileTreeModel::emitParentsChanged(QModelIndex const& index, int firstColumn, int lastColumn,
QSet<QModelIndex>* visitedParentIndices)
{
assert(firstColumn <= lastColumn);
QModelIndex walk = index;
for (;;)
{
walk = parent(walk, firstColumn);
if (!walk.isValid())
{
break;
}
if (visitedParentIndices != nullptr)
{
if (visitedParentIndices->contains(walk))
{
break;
}
visitedParentIndices->insert(walk);
}
emit dataChanged(walk, walk.sibling(walk.row(), lastColumn));
}
}
void FileTreeModel::emitSubtreeChanged(QModelIndex const& index, int firstColumn, int lastColumn)
{
assert(firstColumn <= lastColumn);
int const childCount = rowCount(index);
if (childCount == 0)
{
return;
}
// tell everyone that this item changed
emit dataChanged(index.child(0, firstColumn), index.child(childCount - 1, lastColumn));
// walk the subitems
for (int i = 0; i < childCount; ++i)
{
emitSubtreeChanged(index.child(i, 0), firstColumn, lastColumn);
}
}
void FileTreeModel::twiddleWanted(QModelIndexList const& indices)
{
QMap<bool, QModelIndexList> wantedIndices;
for (QModelIndex const& i : getOrphanIndices(indices))
{
FileTreeItem const* const item = itemFromIndex(i);
wantedIndices[item->isSubtreeWanted() != Qt::Checked] << i;
}
for (int i = 0; i <= 1; ++i)
{
if (wantedIndices.contains(i))
{
setWanted(wantedIndices[i], i != 0);
}
}
}
void FileTreeModel::twiddlePriority(QModelIndexList const& indices)
{
QMap<int, QModelIndexList> priorityIndices;
for (QModelIndex const& i : getOrphanIndices(indices))
{
FileTreeItem const* const item = itemFromIndex(i);
int priority = item->priority();
// ... -> normal -> high -> low -> normal -> ...; mixed -> normal
if (priority == FileTreeItem::NORMAL)
{
priority = TR_PRI_HIGH;
}
else if (priority == FileTreeItem::HIGH)
{
priority = TR_PRI_LOW;
}
else
{
priority = TR_PRI_NORMAL;
}
priorityIndices[priority] << i;
}
for (int i = TR_PRI_LOW; i <= TR_PRI_HIGH; ++i)
{
if (priorityIndices.contains(i))
{
setPriority(priorityIndices[i], i);
}
}
}
void FileTreeModel::setWanted(QModelIndexList const& indices, bool wanted)
{
if (indices.isEmpty())
{
return;
}
QModelIndexList const orphanIndices = getOrphanIndices(indices);
QSet<int> fileIds;
for (QModelIndex const& i : orphanIndices)
{
FileTreeItem* const item = itemFromIndex(i);
item->setSubtreeWanted(wanted, fileIds);
emit dataChanged(i, i);
emitSubtreeChanged(i, COL_WANTED, COL_WANTED);
}
// emit parent changes separately to avoid multiple updates for same items
QSet<QModelIndex> parentIndices;
for (QModelIndex const& i : orphanIndices)
{
emitParentsChanged(i, COL_SIZE, COL_WANTED, &parentIndices);
}
if (!fileIds.isEmpty())
{
emit wantedChanged(fileIds, wanted);
}
}
void FileTreeModel::setPriority(QModelIndexList const& indices, int priority)
{
if (indices.isEmpty())
{
return;
}
QModelIndexList const orphanIndices = getOrphanIndices(indices);
QSet<int> fileIds;
for (QModelIndex const& i : orphanIndices)
{
FileTreeItem* const item = itemFromIndex(i);
item->setSubtreePriority(priority, fileIds);
emit dataChanged(i, i);
emitSubtreeChanged(i, COL_PRIORITY, COL_PRIORITY);
}
// emit parent changes separately to avoid multiple updates for same items
QSet<QModelIndex> parentIndices;
for (QModelIndex const& i : orphanIndices)
{
emitParentsChanged(i, COL_PRIORITY, COL_PRIORITY, &parentIndices);
}
if (!fileIds.isEmpty())
{
emit priorityChanged(fileIds, priority);
}
}
bool FileTreeModel::openFile(QModelIndex const& index)
{
if (!index.isValid())
{
return false;
}
FileTreeItem* const item = itemFromIndex(index);
if (item->fileIndex() < 0 || !item->isComplete())
{
return false;
}
emit openRequested(item->path());
return true;
}