487 lines
11 KiB
C++
487 lines
11 KiB
C++
// This file Copyright © Mnemosyne LLC.
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
// License text can be found in the licenses/ folder.
|
|
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <cstddef>
|
|
#include <ctime>
|
|
#include <iterator> // for std::back_inserter
|
|
#include <set>
|
|
#include <string_view>
|
|
#include <vector>
|
|
|
|
#include <libtransmission/transmission.h>
|
|
|
|
#include <libtransmission/quark.h>
|
|
#include <libtransmission/variant.h>
|
|
|
|
#include "Torrent.h"
|
|
#include "TorrentDelegate.h"
|
|
#include "TorrentModel.h"
|
|
#include "VariantHelpers.h"
|
|
|
|
using ::trqt::variant_helpers::getValue;
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
namespace
|
|
{
|
|
|
|
struct TorrentIdLessThan
|
|
{
|
|
bool operator()(Torrent const* left, Torrent const* right) const
|
|
{
|
|
return left->id() < right->id();
|
|
}
|
|
|
|
bool operator()(int left_id, Torrent const* right) const
|
|
{
|
|
return left_id < right->id();
|
|
}
|
|
|
|
bool operator()(Torrent const* left, int right_id) const
|
|
{
|
|
return left->id() < right_id;
|
|
}
|
|
};
|
|
|
|
template<typename Iter>
|
|
auto getIds(Iter it, Iter end)
|
|
{
|
|
torrent_ids_t ids;
|
|
|
|
for (; it != end; ++it)
|
|
{
|
|
ids.insert((*it)->id());
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
TorrentModel::TorrentModel(Prefs const& prefs)
|
|
: prefs_{ prefs }
|
|
{
|
|
}
|
|
|
|
TorrentModel::~TorrentModel()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
void TorrentModel::clear()
|
|
{
|
|
beginResetModel();
|
|
qDeleteAll(torrents_);
|
|
torrents_.clear();
|
|
endResetModel();
|
|
}
|
|
|
|
int TorrentModel::rowCount(QModelIndex const& parent) const
|
|
{
|
|
Q_UNUSED(parent)
|
|
|
|
return torrents_.size();
|
|
}
|
|
|
|
QVariant TorrentModel::data(QModelIndex const& index, int role) const
|
|
{
|
|
auto const* const t = (index.isValid() && index.row() < rowCount()) ? torrents_.at(index.row()) : nullptr;
|
|
|
|
if (t != nullptr)
|
|
{
|
|
switch (role)
|
|
{
|
|
case Qt::DisplayRole:
|
|
return t->name();
|
|
|
|
case Qt::DecorationRole:
|
|
return t->getMimeTypeIcon();
|
|
|
|
case TorrentRole:
|
|
return QVariant::fromValue(t);
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
void TorrentModel::removeTorrents(tr_variant* torrent_list)
|
|
{
|
|
auto torrents = torrents_t{};
|
|
torrents.reserve(tr_variantListSize(torrent_list));
|
|
|
|
int i = 0;
|
|
tr_variant const* child = nullptr;
|
|
while ((child = tr_variantListChild(torrent_list, i++)) != nullptr)
|
|
{
|
|
if (auto const id = getValue<int>(child); id)
|
|
{
|
|
if (auto* const torrent = getTorrentFromId(*id); torrent != nullptr)
|
|
{
|
|
torrents.push_back(torrent);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!torrents.empty())
|
|
{
|
|
rowsRemove(torrents);
|
|
}
|
|
}
|
|
|
|
void TorrentModel::updateTorrents(tr_variant* torrent_list, bool is_complete_list)
|
|
{
|
|
auto const old = is_complete_list ? torrents_ : torrents_t{};
|
|
auto added = torrent_ids_t{};
|
|
auto changed = torrent_ids_t{};
|
|
auto completed = torrent_ids_t{};
|
|
auto edited = torrent_ids_t{};
|
|
auto instantiated = torrents_t{};
|
|
auto needinfo = torrent_ids_t{};
|
|
auto processed = torrents_t{};
|
|
auto changed_fields = Torrent::fields_t{};
|
|
|
|
auto const now = time(nullptr);
|
|
auto const recently_added = [&now](auto const& tor)
|
|
{
|
|
static auto constexpr MaxAge = 60;
|
|
auto const date = tor->dateAdded();
|
|
return (date != 0) && (difftime(now, date) < MaxAge);
|
|
};
|
|
|
|
// build a list of the property keys
|
|
tr_variant* const first_child = tr_variantListChild(torrent_list, 0);
|
|
bool const table = first_child != nullptr && first_child->holds_alternative<tr_variant::Vector>();
|
|
std::vector<tr_quark> keys;
|
|
if (table)
|
|
{
|
|
// In 'table' format, the first entry in 'torrents' is an array of keys.
|
|
// All the other entries are an array of the values for one torrent.
|
|
auto sv = std::string_view{};
|
|
size_t i = 0;
|
|
keys.reserve(tr_variantListSize(first_child));
|
|
while (tr_variantGetStrView(tr_variantListChild(first_child, i++), &sv))
|
|
{
|
|
keys.push_back(tr_quark_new(sv));
|
|
}
|
|
}
|
|
else if (first_child != nullptr)
|
|
{
|
|
// In 'object' format, every entry is an object with the same set of properties
|
|
auto key = tr_quark{};
|
|
tr_variant* value = nullptr;
|
|
for (size_t i = 0; tr_variantDictChild(first_child, i, &key, &value); ++i)
|
|
{
|
|
keys.push_back(key);
|
|
}
|
|
}
|
|
|
|
// Find the position of TR_KEY_id so we can do torrent lookup
|
|
auto const id_it = std::find(std::begin(keys), std::end(keys), TR_KEY_id);
|
|
if (id_it == std::end(keys)) // no ids provided; we can't proceed
|
|
{
|
|
return;
|
|
}
|
|
|
|
auto const id_pos = std::distance(std::begin(keys), id_it);
|
|
|
|
// Loop through the torrent records...
|
|
std::vector<tr_variant*> values;
|
|
values.reserve(keys.size());
|
|
size_t tor_index = table ? 1 : 0;
|
|
processed.reserve(tr_variantListSize(torrent_list));
|
|
tr_variant* v = nullptr;
|
|
while ((v = tr_variantListChild(torrent_list, tor_index++)))
|
|
{
|
|
// Build an array of values
|
|
values.clear();
|
|
if (table)
|
|
{
|
|
// In table mode, v is already a list of values
|
|
size_t i = 0;
|
|
tr_variant* val = nullptr;
|
|
while ((val = tr_variantListChild(v, i++)))
|
|
{
|
|
values.push_back(val);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// In object mode, v is an object of torrent property key/vals
|
|
size_t i = 0;
|
|
auto key = tr_quark{};
|
|
tr_variant* value = nullptr;
|
|
while (tr_variantDictChild(v, i++, &key, &value))
|
|
{
|
|
values.push_back(value);
|
|
}
|
|
}
|
|
|
|
// Find the torrent id
|
|
auto const id = getValue<int>(values[id_pos]);
|
|
if (!id)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
Torrent* tor = getTorrentFromId(*id);
|
|
bool is_new = false;
|
|
|
|
if (tor == nullptr)
|
|
{
|
|
tor = new Torrent{ prefs_, *id };
|
|
instantiated.push_back(tor);
|
|
is_new = true;
|
|
}
|
|
|
|
auto const fields = tor->update(keys.data(), values.data(), keys.size());
|
|
|
|
if (fields.any())
|
|
{
|
|
changed_fields |= fields;
|
|
changed.insert(*id);
|
|
}
|
|
|
|
if (fields.test(Torrent::EDIT_DATE))
|
|
{
|
|
edited.insert(*id);
|
|
}
|
|
|
|
if (is_new && !tor->hasName())
|
|
{
|
|
needinfo.insert(*id);
|
|
}
|
|
|
|
if (recently_added(tor) && tor->hasName() && !already_added_.count(*id))
|
|
{
|
|
added.insert(*id);
|
|
already_added_.insert(*id);
|
|
}
|
|
|
|
if (fields.test(Torrent::LEFT_UNTIL_DONE) && (tor->leftUntilDone() == 0) && (tor->downloadedEver() > 0))
|
|
{
|
|
completed.insert(*id);
|
|
}
|
|
|
|
processed.push_back(tor);
|
|
}
|
|
|
|
// model upkeep
|
|
|
|
if (!instantiated.empty())
|
|
{
|
|
rowsAdd(instantiated);
|
|
}
|
|
|
|
if (!edited.empty())
|
|
{
|
|
emit torrentsEdited(edited);
|
|
}
|
|
|
|
if (!changed.empty())
|
|
{
|
|
rowsEmitChanged(changed);
|
|
}
|
|
|
|
// emit signals
|
|
|
|
if (!added.empty())
|
|
{
|
|
emit torrentsAdded(added);
|
|
}
|
|
|
|
if (!needinfo.empty())
|
|
{
|
|
emit torrentsNeedInfo(needinfo);
|
|
}
|
|
|
|
if (!changed.empty())
|
|
{
|
|
emit torrentsChanged(changed, changed_fields);
|
|
}
|
|
|
|
if (!completed.empty())
|
|
{
|
|
emit torrentsCompleted(completed);
|
|
}
|
|
|
|
// model upkeep
|
|
|
|
if (is_complete_list)
|
|
{
|
|
std::sort(processed.begin(), processed.end(), TorrentIdLessThan());
|
|
torrents_t removed;
|
|
removed.reserve(old.size());
|
|
std::set_difference(old.begin(), old.end(), processed.begin(), processed.end(), std::back_inserter(removed));
|
|
rowsRemove(removed);
|
|
}
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
std::optional<int> TorrentModel::getRow(int id) const
|
|
{
|
|
std::optional<int> row;
|
|
|
|
auto const [begin, end] = std::equal_range(torrents_.begin(), torrents_.end(), id, TorrentIdLessThan());
|
|
if (begin != end)
|
|
{
|
|
row = std::distance(torrents_.begin(), begin);
|
|
assert(torrents_[*row]->id() == id);
|
|
}
|
|
|
|
return row;
|
|
}
|
|
|
|
Torrent* TorrentModel::getTorrentFromId(int id)
|
|
{
|
|
auto const row = getRow(id);
|
|
return row ? torrents_[*row] : nullptr;
|
|
}
|
|
|
|
Torrent const* TorrentModel::getTorrentFromId(int id) const
|
|
{
|
|
auto const row = getRow(id);
|
|
return row ? torrents_[*row] : nullptr;
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
std::vector<TorrentModel::span_t> TorrentModel::getSpans(torrent_ids_t const& ids) const
|
|
{
|
|
// ids -> rows
|
|
std::vector<int> rows;
|
|
rows.reserve(ids.size());
|
|
for (auto const& id : ids)
|
|
{
|
|
auto const row = getRow(id);
|
|
if (row)
|
|
{
|
|
rows.push_back(*row);
|
|
}
|
|
}
|
|
|
|
std::sort(rows.begin(), rows.end());
|
|
|
|
// rows -> spans
|
|
std::vector<span_t> spans;
|
|
spans.reserve(rows.size());
|
|
span_t span;
|
|
bool in_span = false;
|
|
for (auto const& row : rows)
|
|
{
|
|
if (in_span)
|
|
{
|
|
if (span.second + 1 == row)
|
|
{
|
|
span.second = row;
|
|
}
|
|
else
|
|
{
|
|
spans.push_back(span);
|
|
in_span = false;
|
|
}
|
|
}
|
|
|
|
if (!in_span)
|
|
{
|
|
span.first = span.second = row;
|
|
in_span = true;
|
|
}
|
|
}
|
|
|
|
if (in_span)
|
|
{
|
|
spans.push_back(span);
|
|
}
|
|
|
|
return spans;
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
void TorrentModel::rowsEmitChanged(torrent_ids_t const& ids)
|
|
{
|
|
for (auto const& [first, last] : getSpans(ids))
|
|
{
|
|
emit dataChanged(index(first), index(last));
|
|
}
|
|
}
|
|
|
|
void TorrentModel::rowsAdd(torrents_t const& torrents)
|
|
{
|
|
auto const compare = TorrentIdLessThan();
|
|
|
|
if (torrents_.empty())
|
|
{
|
|
beginInsertRows(QModelIndex{}, 0, torrents.size() - 1);
|
|
torrents_ = torrents;
|
|
std::sort(torrents_.begin(), torrents_.end(), TorrentIdLessThan{});
|
|
endInsertRows();
|
|
}
|
|
else
|
|
{
|
|
for (auto const& tor : torrents)
|
|
{
|
|
auto const it = std::lower_bound(torrents_.begin(), torrents_.end(), tor, compare);
|
|
auto const row = static_cast<int>(std::distance(torrents_.begin(), it));
|
|
|
|
beginInsertRows(QModelIndex{}, row, row);
|
|
torrents_.insert(it, tor);
|
|
endInsertRows();
|
|
}
|
|
}
|
|
}
|
|
|
|
void TorrentModel::rowsRemove(torrents_t const& torrents)
|
|
{
|
|
// must walk in reverse to avoid invalidating row numbers
|
|
auto const& spans = getSpans(getIds(torrents.begin(), torrents.end()));
|
|
for (auto it = spans.rbegin(), end = spans.rend(); it != end; ++it)
|
|
{
|
|
auto const& [first, last] = *it;
|
|
|
|
beginRemoveRows(QModelIndex{}, first, last);
|
|
torrents_.erase(torrents_.begin() + first, torrents_.begin() + last + 1);
|
|
endRemoveRows();
|
|
}
|
|
|
|
qDeleteAll(torrents);
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
bool TorrentModel::hasTorrent(TorrentHash const& hash) const
|
|
{
|
|
auto test = [hash](auto const& tor)
|
|
{
|
|
return tor->hash() == hash;
|
|
};
|
|
return std::any_of(torrents_.cbegin(), torrents_.cend(), test);
|
|
}
|