mirror of
https://github.com/d47081/qBittorrent.git
synced 2025-01-12 07:48:04 +00:00
Merge pull request #3908 from Chocobo1/rm_nsort
Replace `naturalSort()` with `naturalCompare()`.
This commit is contained in:
commit
9c5a5fc83d
@ -27,124 +27,65 @@
|
|||||||
* exception statement from your version.
|
* exception statement from your version.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <QByteArray>
|
|
||||||
#include <QString>
|
|
||||||
#include <QLocale>
|
|
||||||
#include <cmath>
|
|
||||||
#include "string.h"
|
#include "string.h"
|
||||||
|
|
||||||
QString Utils::String::fromStdString(const std::string &str)
|
#include <cmath>
|
||||||
|
|
||||||
|
#include <QByteArray>
|
||||||
|
#include <QtGlobal>
|
||||||
|
#include <QLocale>
|
||||||
|
#ifdef QBT_USES_QT5
|
||||||
|
#include <QCollator>
|
||||||
|
#endif
|
||||||
|
#ifdef Q_OS_MAC
|
||||||
|
#include <QThreadStorage>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace
|
||||||
{
|
{
|
||||||
return QString::fromUtf8(str.c_str());
|
class NaturalCompare
|
||||||
}
|
{
|
||||||
|
public:
|
||||||
std::string Utils::String::toStdString(const QString &str)
|
explicit NaturalCompare(const bool caseSensitive = true)
|
||||||
{
|
: m_caseSensitive(caseSensitive)
|
||||||
QByteArray utf8 = str.toUtf8();
|
{
|
||||||
return std::string(utf8.constData(), utf8.length());
|
#ifdef QBT_USES_QT5
|
||||||
}
|
|
||||||
|
|
||||||
// uses lessThan comparison
|
|
||||||
bool Utils::String::naturalSort(const QString &left, const QString &right, bool &result)
|
|
||||||
{
|
|
||||||
// Return value indicates if functions was successful
|
|
||||||
// result argument will contain actual comparison result if function was successful
|
|
||||||
int posL = 0;
|
|
||||||
int posR = 0;
|
|
||||||
do {
|
|
||||||
forever {
|
|
||||||
if (posL == left.size() || posR == right.size())
|
|
||||||
return false; // No data
|
|
||||||
|
|
||||||
QChar leftChar = left.at(posL);
|
|
||||||
QChar rightChar = right.at(posR);
|
|
||||||
bool leftCharIsDigit = leftChar.isDigit();
|
|
||||||
bool rightCharIsDigit = rightChar.isDigit();
|
|
||||||
if (leftCharIsDigit != rightCharIsDigit)
|
|
||||||
return false; // Digit positions mismatch
|
|
||||||
|
|
||||||
if (leftCharIsDigit)
|
|
||||||
break; // Both are digit, break this loop and compare numbers
|
|
||||||
|
|
||||||
if (leftChar != rightChar)
|
|
||||||
return false; // Strings' subsets before digit do not match
|
|
||||||
|
|
||||||
++posL;
|
|
||||||
++posR;
|
|
||||||
}
|
|
||||||
|
|
||||||
QString temp;
|
|
||||||
while (posL < left.size()) {
|
|
||||||
if (left.at(posL).isDigit())
|
|
||||||
temp += left.at(posL);
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
posL++;
|
|
||||||
}
|
|
||||||
int numL = temp.toInt();
|
|
||||||
temp.clear();
|
|
||||||
|
|
||||||
while (posR < right.size()) {
|
|
||||||
if (right.at(posR).isDigit())
|
|
||||||
temp += right.at(posR);
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
posR++;
|
|
||||||
}
|
|
||||||
int numR = temp.toInt();
|
|
||||||
|
|
||||||
if (numL != numR) {
|
|
||||||
result = (numL < numR);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Strings + digits do match and we haven't hit string end
|
|
||||||
// Do another round
|
|
||||||
|
|
||||||
} while (true);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Utils::String::NaturalCompare::NaturalCompare()
|
|
||||||
{
|
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
|
|
||||||
#if defined(Q_OS_WIN)
|
#if defined(Q_OS_WIN)
|
||||||
// Without ICU library, QCollator doesn't support `setNumericMode(true)` on OS older than Win7
|
// Without ICU library, QCollator doesn't support `setNumericMode(true)` on OS older than Win7
|
||||||
if(QSysInfo::windowsVersion() < QSysInfo::WV_WINDOWS7)
|
if (QSysInfo::windowsVersion() < QSysInfo::WV_WINDOWS7)
|
||||||
return;
|
return;
|
||||||
#endif
|
#endif
|
||||||
m_collator.setNumericMode(true);
|
m_collator.setNumericMode(true);
|
||||||
m_collator.setCaseSensitivity(Qt::CaseInsensitive);
|
m_collator.setCaseSensitivity(caseSensitive ? Qt::CaseSensitive : Qt::CaseInsensitive);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Utils::String::NaturalCompare::operator()(const QString &l, const QString &r)
|
bool operator()(const QString &left, const QString &right) const
|
||||||
{
|
{
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
|
#ifdef QBT_USES_QT5
|
||||||
#if defined(Q_OS_WIN)
|
#if defined(Q_OS_WIN)
|
||||||
// Without ICU library, QCollator doesn't support `setNumericMode(true)` on OS older than Win7
|
// Without ICU library, QCollator doesn't support `setNumericMode(true)` on OS older than Win7
|
||||||
if(QSysInfo::windowsVersion() < QSysInfo::WV_WINDOWS7)
|
if (QSysInfo::windowsVersion() < QSysInfo::WV_WINDOWS7)
|
||||||
return lessThan(l, r);
|
return lessThan(left, right);
|
||||||
#endif
|
#endif
|
||||||
return (m_collator.compare(l, r) < 0);
|
return (m_collator.compare(left, right) < 0);
|
||||||
#else
|
#else
|
||||||
return lessThan(l, r);
|
return lessThan(left, right);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Utils::String::NaturalCompare::lessThan(const QString &left, const QString &right)
|
bool lessThan(const QString &left, const QString &right) const
|
||||||
{
|
{
|
||||||
// Return value `false` indicates `right` should go before `left`, otherwise, after
|
// Return value `false` indicates `right` should go before `left`, otherwise, after
|
||||||
int posL = 0;
|
int posL = 0;
|
||||||
int posR = 0;
|
int posR = 0;
|
||||||
while (true) {
|
while (true) {
|
||||||
while (true) {
|
while (true) {
|
||||||
if (posL == left.size() || posR == right.size())
|
if ((posL == left.size()) || (posR == right.size()))
|
||||||
return (left.size() < right.size()); // when a shorter string is another string's prefix, shorter string place before longer string
|
return (left.size() < right.size()); // when a shorter string is another string's prefix, shorter string place before longer string
|
||||||
|
|
||||||
QChar leftChar = left[posL].toLower();
|
QChar leftChar = m_caseSensitive ? left[posL] : left[posL].toLower();
|
||||||
QChar rightChar = right[posR].toLower();
|
QChar rightChar = m_caseSensitive ? right[posR] : right[posR].toLower();
|
||||||
if (leftChar == rightChar)
|
if (leftChar == rightChar)
|
||||||
; // compare next character
|
; // compare next character
|
||||||
else if (leftChar.isDigit() && rightChar.isDigit())
|
else if (leftChar.isDigit() && rightChar.isDigit())
|
||||||
@ -159,7 +100,7 @@ bool Utils::String::NaturalCompare::lessThan(const QString &left, const QString
|
|||||||
int startL = posL;
|
int startL = posL;
|
||||||
while ((posL < left.size()) && left[posL].isDigit())
|
while ((posL < left.size()) && left[posL].isDigit())
|
||||||
++posL;
|
++posL;
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5, 1, 0)
|
#ifdef QBT_USES_QT5
|
||||||
int numL = left.midRef(startL, posL - startL).toInt();
|
int numL = left.midRef(startL, posL - startL).toInt();
|
||||||
#else
|
#else
|
||||||
int numL = left.mid(startL, posL - startL).toInt();
|
int numL = left.mid(startL, posL - startL).toInt();
|
||||||
@ -168,7 +109,7 @@ bool Utils::String::NaturalCompare::lessThan(const QString &left, const QString
|
|||||||
int startR = posR;
|
int startR = posR;
|
||||||
while ((posR < right.size()) && right[posR].isDigit())
|
while ((posR < right.size()) && right[posR].isDigit())
|
||||||
++posR;
|
++posR;
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5, 1, 0)
|
#ifdef QBT_USES_QT5
|
||||||
int numR = right.midRef(startR, posR - startR).toInt();
|
int numR = right.midRef(startR, posR - startR).toInt();
|
||||||
#else
|
#else
|
||||||
int numR = right.mid(startR, posR - startR).toInt();
|
int numR = right.mid(startR, posR - startR).toInt();
|
||||||
@ -181,6 +122,53 @@ bool Utils::String::NaturalCompare::lessThan(const QString &left, const QString
|
|||||||
// Do another round
|
// Do another round
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
#ifdef QBT_USES_QT5
|
||||||
|
QCollator m_collator;
|
||||||
|
#endif
|
||||||
|
const bool m_caseSensitive;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Utils::String::naturalCompareCaseSensitive(const QString &left, const QString &right)
|
||||||
|
{
|
||||||
|
// provide a single `NaturalCompare` instance for easy use
|
||||||
|
// https://doc.qt.io/qt-5/threads-reentrancy.html
|
||||||
|
#ifdef Q_OS_MAC // workaround for Apple xcode: https://stackoverflow.com/a/29929949
|
||||||
|
static QThreadStorage<NaturalCompare> nCmp;
|
||||||
|
if (!nCmp.hasLocalData()) nCmp.setLocalData(NaturalCompare(true));
|
||||||
|
return (nCmp.localData())(left, right);
|
||||||
|
#else
|
||||||
|
thread_local NaturalCompare nCmp(true);
|
||||||
|
return nCmp(left, right);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Utils::String::naturalCompareCaseInsensitive(const QString &left, const QString &right)
|
||||||
|
{
|
||||||
|
// provide a single `NaturalCompare` instance for easy use
|
||||||
|
// https://doc.qt.io/qt-5/threads-reentrancy.html
|
||||||
|
#ifdef Q_OS_MAC // workaround for Apple xcode: https://stackoverflow.com/a/29929949
|
||||||
|
static QThreadStorage<NaturalCompare> nCmp;
|
||||||
|
if (!nCmp.hasLocalData()) nCmp.setLocalData(NaturalCompare(false));
|
||||||
|
return (nCmp.localData())(left, right);
|
||||||
|
#else
|
||||||
|
thread_local NaturalCompare nCmp(false);
|
||||||
|
return nCmp(left, right);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
QString Utils::String::fromStdString(const std::string &str)
|
||||||
|
{
|
||||||
|
return QString::fromUtf8(str.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Utils::String::toStdString(const QString &str)
|
||||||
|
{
|
||||||
|
QByteArray utf8 = str.toUtf8();
|
||||||
|
return std::string(utf8.constData(), utf8.length());
|
||||||
}
|
}
|
||||||
|
|
||||||
// to send numbers instead of strings with suffixes
|
// to send numbers instead of strings with suffixes
|
||||||
|
@ -31,13 +31,9 @@
|
|||||||
#define UTILS_STRING_H
|
#define UTILS_STRING_H
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <QtGlobal>
|
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
|
|
||||||
#include <QCollator>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
class QString;
|
|
||||||
class QByteArray;
|
class QByteArray;
|
||||||
|
class QString;
|
||||||
|
|
||||||
namespace Utils
|
namespace Utils
|
||||||
{
|
{
|
||||||
@ -51,19 +47,8 @@ namespace Utils
|
|||||||
// Taken from https://crackstation.net/hashing-security.htm
|
// Taken from https://crackstation.net/hashing-security.htm
|
||||||
bool slowEquals(const QByteArray &a, const QByteArray &b);
|
bool slowEquals(const QByteArray &a, const QByteArray &b);
|
||||||
|
|
||||||
bool naturalSort(const QString &left, const QString &right, bool &result);
|
bool naturalCompareCaseSensitive(const QString &left, const QString &right);
|
||||||
|
bool naturalCompareCaseInsensitive(const QString &left, const QString &right);
|
||||||
class NaturalCompare
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
NaturalCompare();
|
|
||||||
bool operator()(const QString &l, const QString &r);
|
|
||||||
bool lessThan(const QString &left, const QString &right);
|
|
||||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
|
|
||||||
private:
|
|
||||||
QCollator m_collator;
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,7 +96,7 @@ AddNewTorrentDialog::AddNewTorrentDialog(QWidget *parent)
|
|||||||
|
|
||||||
// Load categories
|
// Load categories
|
||||||
QStringList categories = session->categories();
|
QStringList categories = session->categories();
|
||||||
std::sort(categories.begin(), categories.end(), Utils::String::NaturalCompare());
|
std::sort(categories.begin(), categories.end(), Utils::String::naturalCompareCaseInsensitive);
|
||||||
QString defaultCategory = settings()->loadValue(KEY_DEFAULTCATEGORY).toString();
|
QString defaultCategory = settings()->loadValue(KEY_DEFAULTCATEGORY).toString();
|
||||||
|
|
||||||
if (!defaultCategory.isEmpty())
|
if (!defaultCategory.isEmpty())
|
||||||
|
@ -43,20 +43,15 @@ public:
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool lessThan(const QModelIndex &left, const QModelIndex &right) const {
|
bool lessThan(const QModelIndex &left, const QModelIndex &right) const {
|
||||||
if (sortColumn() == PeerListDelegate::IP || sortColumn() == PeerListDelegate::CLIENT) {
|
switch (sortColumn()) {
|
||||||
QVariant vL = sourceModel()->data(left);
|
case PeerListDelegate::IP:
|
||||||
QVariant vR = sourceModel()->data(right);
|
case PeerListDelegate::CLIENT: {
|
||||||
if (!(vL.isValid() && vR.isValid()))
|
QString vL = left.data().toString();
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
QString vR = right.data().toString();
|
||||||
Q_ASSERT(vL.isValid());
|
return Utils::String::naturalCompareCaseSensitive(vL, vR);
|
||||||
Q_ASSERT(vR.isValid());
|
|
||||||
|
|
||||||
bool res = false;
|
|
||||||
if (Utils::String::naturalSort(vL.toString(), vR.toString(), res))
|
|
||||||
return res;
|
|
||||||
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
return QSortFilterProxyModel::lessThan(left, right);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -314,7 +314,7 @@ void AutomatedRssDownloader::initCategoryCombobox()
|
|||||||
{
|
{
|
||||||
// Load torrent categories
|
// Load torrent categories
|
||||||
QStringList categories = BitTorrent::Session::instance()->categories();
|
QStringList categories = BitTorrent::Session::instance()->categories();
|
||||||
std::sort(categories.begin(), categories.end(), Utils::String::NaturalCompare());
|
std::sort(categories.begin(), categories.end(), Utils::String::naturalCompareCaseInsensitive);
|
||||||
ui->comboCategory->addItems(categories);
|
ui->comboCategory->addItems(categories);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,20 +107,17 @@ qint64 SearchSortModel::maxSize() const
|
|||||||
|
|
||||||
bool SearchSortModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
bool SearchSortModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
||||||
{
|
{
|
||||||
if ((sortColumn() == NAME) || (sortColumn() == ENGINE_URL)) {
|
switch (sortColumn()) {
|
||||||
QVariant vL = sourceModel()->data(left);
|
case NAME:
|
||||||
QVariant vR = sourceModel()->data(right);
|
case ENGINE_URL: {
|
||||||
if (!(vL.isValid() && vR.isValid()))
|
QString vL = left.data().toString();
|
||||||
return base::lessThan(left, right);
|
QString vR = right.data().toString();
|
||||||
Q_ASSERT(vL.isValid());
|
return Utils::String::naturalCompareCaseSensitive(vL, vR);
|
||||||
Q_ASSERT(vR.isValid());
|
|
||||||
|
|
||||||
bool res = false;
|
|
||||||
if (Utils::String::naturalSort(vL.toString(), vR.toString(), res))
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
return base::lessThan(left, right);
|
return base::lessThan(left, right);
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SearchSortModel::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const
|
bool SearchSortModel::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const
|
||||||
|
@ -82,28 +82,22 @@ bool TorrentContentFilterModel::filterAcceptsRow(int source_row, const QModelInd
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool TorrentContentFilterModel::lessThan(const QModelIndex &left, const QModelIndex &right) const {
|
bool TorrentContentFilterModel::lessThan(const QModelIndex &left, const QModelIndex &right) const {
|
||||||
if (sortColumn() == NAME) {
|
switch (sortColumn()) {
|
||||||
QVariant vL = sourceModel()->data(left);
|
case NAME: { // PropColumn::NAME
|
||||||
QVariant vR = sourceModel()->data(right);
|
QString vL = left.data().toString();
|
||||||
if (!(vL.isValid() && vR.isValid()))
|
QString vR = right.data().toString();
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
TorrentContentModelItem::ItemType leftType = m_model->itemType(m_model->index(left.row(), 0, left.parent()));
|
||||||
Q_ASSERT(vL.isValid());
|
TorrentContentModelItem::ItemType rightType = m_model->itemType(m_model->index(right.row(), 0, right.parent()));
|
||||||
Q_ASSERT(vR.isValid());
|
|
||||||
|
|
||||||
TorrentContentModelItem::ItemType leftType, rightType;
|
if (leftType == rightType)
|
||||||
leftType = m_model->itemType(m_model->index(left.row(), 0, left.parent()));
|
return Utils::String::naturalCompareCaseSensitive(vL, vR);
|
||||||
rightType = m_model->itemType(m_model->index(right.row(), 0, right.parent()));
|
|
||||||
if (leftType == rightType) {
|
|
||||||
bool res = false;
|
|
||||||
if (Utils::String::naturalSort(vL.toString(), vR.toString(), res))
|
|
||||||
return res;
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
|
||||||
}
|
|
||||||
else if (leftType == TorrentContentModelItem::FolderType && sortOrder() == Qt::AscendingOrder)
|
else if (leftType == TorrentContentModelItem::FolderType && sortOrder() == Qt::AscendingOrder)
|
||||||
return true;
|
return true;
|
||||||
else
|
else
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
return QSortFilterProxyModel::lessThan(left, right);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -237,17 +237,14 @@ void CategoryFiltersList::addItem(const QString &category, bool hasTorrent)
|
|||||||
if (exists) return;
|
if (exists) return;
|
||||||
|
|
||||||
Q_ASSERT(count() >= 2);
|
Q_ASSERT(count() >= 2);
|
||||||
|
int insPos = count();
|
||||||
for (int i = 2; i < count(); ++i) {
|
for (int i = 2; i < count(); ++i) {
|
||||||
bool less = false;
|
if (Utils::String::naturalCompareCaseSensitive(category, item(i)->text())) {
|
||||||
if (!(Utils::String::naturalSort(category, item(i)->text(), less)))
|
insPos = i;
|
||||||
less = (category.localeAwareCompare(item(i)->text()) < 0);
|
break;
|
||||||
if (less) {
|
|
||||||
insertItem(i, categoryItem);
|
|
||||||
updateGeometry();
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
QListWidget::addItem(categoryItem);
|
QListWidget::insertItem(insPos, categoryItem);
|
||||||
updateGeometry();
|
updateGeometry();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -512,17 +509,14 @@ void TrackerFiltersList::addItem(const QString &tracker, const QString &hash)
|
|||||||
}
|
}
|
||||||
|
|
||||||
Q_ASSERT(count() >= 4);
|
Q_ASSERT(count() >= 4);
|
||||||
for (int i = 4; i<count(); ++i) {
|
int insPos = count();
|
||||||
bool less = false;
|
for (int i = 4; i < count(); ++i) {
|
||||||
if (!(Utils::String::naturalSort(host, item(i)->text(), less)))
|
if (Utils::String::naturalCompareCaseSensitive(host, item(i)->text())) {
|
||||||
less = (host.localeAwareCompare(item(i)->text()) < 0);
|
insPos = i;
|
||||||
if (less) {
|
break;
|
||||||
insertItem(i, trackerItem);
|
|
||||||
updateGeometry();
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
QListWidget::addItem(trackerItem);
|
QListWidget::insertItem(insPos, trackerItem);
|
||||||
updateGeometry();
|
updateGeometry();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -73,21 +73,19 @@ void TransferListSortModel::disableTrackerFilter()
|
|||||||
|
|
||||||
bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
|
||||||
{
|
{
|
||||||
const int column = sortColumn();
|
switch (sortColumn()) {
|
||||||
|
case TorrentModel::TR_NAME: {
|
||||||
if (column == TorrentModel::TR_NAME) {
|
|
||||||
QVariant vL = left.data();
|
QVariant vL = left.data();
|
||||||
QVariant vR = right.data();
|
QVariant vR = right.data();
|
||||||
if (!vL.isValid() || !vR.isValid() || (vL == vR))
|
if (!vL.isValid() || !vR.isValid() || (vL == vR))
|
||||||
return lowerPositionThan(left, right);
|
return lowerPositionThan(left, right);
|
||||||
|
|
||||||
bool res = false;
|
return Utils::String::naturalCompareCaseSensitive(vL.toString(), vR.toString());
|
||||||
if (Utils::String::naturalSort(vL.toString(), vR.toString(), res))
|
|
||||||
return res;
|
|
||||||
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_ADD_DATE || column == TorrentModel::TR_SEED_DATE || column == TorrentModel::TR_SEEN_COMPLETE_DATE) {
|
|
||||||
|
case TorrentModel::TR_ADD_DATE:
|
||||||
|
case TorrentModel::TR_SEED_DATE:
|
||||||
|
case TorrentModel::TR_SEEN_COMPLETE_DATE: {
|
||||||
QDateTime vL = left.data().toDateTime();
|
QDateTime vL = left.data().toDateTime();
|
||||||
QDateTime vR = right.data().toDateTime();
|
QDateTime vR = right.data().toDateTime();
|
||||||
|
|
||||||
@ -97,10 +95,13 @@ bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex
|
|||||||
|
|
||||||
return vL < vR;
|
return vL < vR;
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_PRIORITY) {
|
|
||||||
|
case TorrentModel::TR_PRIORITY: {
|
||||||
return lowerPositionThan(left, right);
|
return lowerPositionThan(left, right);
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_PEERS || column == TorrentModel::TR_SEEDS) {
|
|
||||||
|
case TorrentModel::TR_SEEDS:
|
||||||
|
case TorrentModel::TR_PEERS: {
|
||||||
int left_active = left.data().toInt();
|
int left_active = left.data().toInt();
|
||||||
int left_total = left.data(Qt::UserRole).toInt();
|
int left_total = left.data(Qt::UserRole).toInt();
|
||||||
int right_active = right.data().toInt();
|
int right_active = right.data().toInt();
|
||||||
@ -116,7 +117,8 @@ bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex
|
|||||||
return (left_active < right_active);
|
return (left_active < right_active);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_ETA) {
|
|
||||||
|
case TorrentModel::TR_ETA: {
|
||||||
TorrentModel *model = qobject_cast<TorrentModel *>(sourceModel());
|
TorrentModel *model = qobject_cast<TorrentModel *>(sourceModel());
|
||||||
const int prioL = model->data(model->index(left.row(), TorrentModel::TR_PRIORITY)).toInt();
|
const int prioL = model->data(model->index(left.row(), TorrentModel::TR_PRIORITY)).toInt();
|
||||||
const int prioR = model->data(model->index(right.row(), TorrentModel::TR_PRIORITY)).toInt();
|
const int prioR = model->data(model->index(right.row(), TorrentModel::TR_PRIORITY)).toInt();
|
||||||
@ -164,7 +166,8 @@ bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex
|
|||||||
return !invalidL;
|
return !invalidL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_LAST_ACTIVITY) {
|
|
||||||
|
case TorrentModel::TR_LAST_ACTIVITY: {
|
||||||
const qlonglong vL = left.data().toLongLong();
|
const qlonglong vL = left.data().toLongLong();
|
||||||
const qlonglong vR = right.data().toLongLong();
|
const qlonglong vR = right.data().toLongLong();
|
||||||
|
|
||||||
@ -173,7 +176,8 @@ bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex
|
|||||||
|
|
||||||
return vL < vR;
|
return vL < vR;
|
||||||
}
|
}
|
||||||
else if (column == TorrentModel::TR_RATIO_LIMIT) {
|
|
||||||
|
case TorrentModel::TR_RATIO_LIMIT: {
|
||||||
const qreal vL = left.data().toDouble();
|
const qreal vL = left.data().toDouble();
|
||||||
const qreal vR = right.data().toDouble();
|
const qreal vR = right.data().toDouble();
|
||||||
|
|
||||||
@ -183,10 +187,12 @@ bool TransferListSortModel::lessThan(const QModelIndex &left, const QModelIndex
|
|||||||
return vL < vR;
|
return vL < vR;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
default: {
|
||||||
if (left.data() == right.data())
|
if (left.data() == right.data())
|
||||||
return lowerPositionThan(left, right);
|
return lowerPositionThan(left, right);
|
||||||
|
|
||||||
return QSortFilterProxyModel::lessThan(left, right);
|
return QSortFilterProxyModel::lessThan(left, right);
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TransferListSortModel::lowerPositionThan(const QModelIndex &left, const QModelIndex &right) const
|
bool TransferListSortModel::lowerPositionThan(const QModelIndex &left, const QModelIndex &right) const
|
||||||
|
@ -769,7 +769,7 @@ void TransferListWidget::displayListMenu(const QPoint&)
|
|||||||
listMenu.addAction(&actionRename);
|
listMenu.addAction(&actionRename);
|
||||||
// Category Menu
|
// Category Menu
|
||||||
QStringList categories = BitTorrent::Session::instance()->categories();
|
QStringList categories = BitTorrent::Session::instance()->categories();
|
||||||
std::sort(categories.begin(), categories.end(), Utils::String::NaturalCompare());
|
std::sort(categories.begin(), categories.end(), Utils::String::naturalCompareCaseInsensitive);
|
||||||
QList<QAction*> categoryActions;
|
QList<QAction*> categoryActions;
|
||||||
QMenu *categoryMenu = listMenu.addMenu(GuiIconProvider::instance()->getIcon("view-categories"), tr("Category"));
|
QMenu *categoryMenu = listMenu.addMenu(GuiIconProvider::instance()->getIcon("view-categories"), tr("Category"));
|
||||||
categoryActions << categoryMenu->addAction(GuiIconProvider::instance()->getIcon("list-add"), tr("New...", "New category..."));
|
categoryActions << categoryMenu->addAction(GuiIconProvider::instance()->getIcon("list-add"), tr("New...", "New category..."));
|
||||||
|
Loading…
Reference in New Issue
Block a user