2023-01-08 15:31:03 +00:00
|
|
|
// This file Copyright © 2022-2023 Mnemosyne LLC.
|
2022-12-21 21:26:25 +00:00
|
|
|
// 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.
|
|
|
|
|
2022-12-26 21:13:21 +00:00
|
|
|
#include "TorrentSorter.h"
|
2022-12-21 21:26:25 +00:00
|
|
|
|
2023-02-18 01:14:01 +00:00
|
|
|
#include "Percents.h"
|
2023-01-08 15:31:03 +00:00
|
|
|
#include "SorterBase.hh"
|
2022-12-29 02:42:20 +00:00
|
|
|
#include "Utils.h"
|
|
|
|
|
2022-12-21 21:26:25 +00:00
|
|
|
#include <libtransmission/transmission.h>
|
2023-07-16 02:55:44 +00:00
|
|
|
#include <libtransmission/utils.h>
|
2022-12-21 21:26:25 +00:00
|
|
|
|
2022-12-26 21:13:21 +00:00
|
|
|
#include <algorithm>
|
2022-12-21 21:26:25 +00:00
|
|
|
|
|
|
|
using namespace std::string_view_literals;
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
constexpr bool is_valid_eta(time_t value)
|
|
|
|
{
|
|
|
|
return value != TR_ETA_NOT_AVAIL && value != TR_ETA_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
constexpr int compare_eta(time_t lhs, time_t rhs)
|
|
|
|
{
|
|
|
|
bool const lhs_valid = is_valid_eta(lhs);
|
|
|
|
bool const rhs_valid = is_valid_eta(rhs);
|
|
|
|
|
|
|
|
if (!lhs_valid && !rhs_valid)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lhs_valid)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!rhs_valid)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return -tr_compare_3way(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
constexpr int compare_ratio(double lhs, double rhs)
|
|
|
|
{
|
|
|
|
if (static_cast<int>(lhs) == TR_RATIO_INF && static_cast<int>(rhs) == TR_RATIO_INF)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (static_cast<int>(lhs) == TR_RATIO_INF)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (static_cast<int>(rhs) == TR_RATIO_INF)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return tr_compare_3way(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_name(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return tr_compare_3way(lhs.get_name_collated(), rhs.get_name_collated());
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_queue(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return tr_compare_3way(lhs.get_queue_position(), rhs.get_queue_position());
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_ratio(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto result = -compare_ratio(lhs.get_ratio(), rhs.get_ratio()); result != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return result;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_queue(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_activity(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_speed_up() + lhs.get_speed_down(), rhs.get_speed_up() + rhs.get_speed_down());
|
|
|
|
val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_active_peer_count(), rhs.get_active_peer_count()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_queue(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_age(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_added_date(), rhs.get_added_date()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_name(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_size(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_total_size(), rhs.get_total_size()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_name(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_progress(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_percent_complete(), rhs.get_percent_complete()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_seed_ratio_percent_done(), rhs.get_seed_ratio_percent_done()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_ratio(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_eta(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = compare_eta(lhs.get_eta(), rhs.get_eta()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_name(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOLINTNEXTLINE(bugprone-easily-swappable-parameters)
|
|
|
|
int compare_by_state(Torrent const& lhs, Torrent const& rhs)
|
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
if (auto val = -tr_compare_3way(lhs.get_activity(), rhs.get_activity()); val != 0)
|
2022-12-21 21:26:25 +00:00
|
|
|
{
|
2023-07-16 02:55:44 +00:00
|
|
|
return val;
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
2023-07-16 02:55:44 +00:00
|
|
|
return compare_by_queue(lhs, rhs);
|
2022-12-21 21:26:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2023-01-08 15:31:03 +00:00
|
|
|
TorrentSorter::TorrentSorter()
|
|
|
|
: Glib::ObjectBase(typeid(TorrentSorter))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-12-21 21:26:25 +00:00
|
|
|
void TorrentSorter::set_mode(std::string_view mode)
|
|
|
|
{
|
|
|
|
static auto const compare_funcs = std::map<std::string_view, CompareFunc>({
|
|
|
|
{ "sort-by-activity"sv, &compare_by_activity },
|
|
|
|
{ "sort-by-age"sv, &compare_by_age },
|
|
|
|
{ "sort-by-name"sv, &compare_by_name },
|
|
|
|
{ "sort-by-progress"sv, &compare_by_progress },
|
|
|
|
{ "sort-by-queue"sv, &compare_by_queue },
|
|
|
|
{ "sort-by-ratio"sv, &compare_by_ratio },
|
|
|
|
{ "sort-by-size"sv, &compare_by_size },
|
|
|
|
{ "sort-by-state"sv, &compare_by_state },
|
|
|
|
{ "sort-by-time-left"sv, &compare_by_eta },
|
|
|
|
});
|
|
|
|
|
|
|
|
auto compare_func = &compare_by_name;
|
|
|
|
if (auto const compare_func_it = compare_funcs.find(mode); compare_func_it != compare_funcs.end())
|
|
|
|
{
|
|
|
|
compare_func = compare_func_it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (compare_func_ == compare_func)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
compare_func_ = compare_func;
|
|
|
|
changed(Change::DIFFERENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentSorter::set_reversed(bool is_reversed)
|
|
|
|
{
|
|
|
|
if (is_reversed_ == is_reversed)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
is_reversed_ = is_reversed;
|
|
|
|
changed(Change::INVERTED);
|
|
|
|
}
|
|
|
|
|
|
|
|
int TorrentSorter::compare(Torrent const& lhs, Torrent const& rhs) const
|
|
|
|
{
|
|
|
|
return compare_func_ != nullptr ? std::clamp(compare_func_(lhs, rhs), -1, 1) * (is_reversed_ ? -1 : 1) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TorrentSorter::update(Torrent::ChangeFlags changes)
|
|
|
|
{
|
|
|
|
using Flag = Torrent::ChangeFlag;
|
|
|
|
|
|
|
|
static auto const compare_flags = std::map<CompareFunc, Torrent::ChangeFlags>({
|
|
|
|
{ &compare_by_activity, Flag::ACTIVE_PEER_COUNT | Flag::QUEUE_POSITION | Flag::SPEED_DOWN | Flag::SPEED_UP },
|
|
|
|
{ &compare_by_age, Flag::ADDED_DATE | Flag::NAME },
|
|
|
|
{ &compare_by_eta, Flag::ETA | Flag::NAME },
|
|
|
|
{ &compare_by_name, Flag::NAME },
|
|
|
|
{ &compare_by_progress, Flag::PERCENT_COMPLETE | Flag::QUEUE_POSITION | Flag::RATIO | Flag::SEED_RATIO_PERCENT_DONE },
|
|
|
|
{ &compare_by_queue, Flag::QUEUE_POSITION },
|
|
|
|
{ &compare_by_ratio, Flag::QUEUE_POSITION | Flag::RATIO },
|
|
|
|
{ &compare_by_size, Flag::NAME | Flag::TOTAL_SIZE },
|
|
|
|
{ &compare_by_state, Flag::ACTIVITY | Flag::QUEUE_POSITION },
|
|
|
|
});
|
|
|
|
|
|
|
|
if (auto const compare_flags_it = compare_flags.find(compare_func_);
|
|
|
|
compare_flags_it != compare_flags.end() && changes.test(compare_flags_it->second))
|
|
|
|
{
|
|
|
|
changed(Change::DIFFERENT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Glib::RefPtr<TorrentSorter> TorrentSorter::create()
|
|
|
|
{
|
2022-12-28 14:47:53 +00:00
|
|
|
// NOLINTNEXTLINE(cppcoreguidelines-owning-memory)
|
2022-12-21 21:26:25 +00:00
|
|
|
return Glib::make_refptr_for_instance(new TorrentSorter());
|
|
|
|
}
|