mirror of
https://github.com/qbittorrent/qBittorrent.git
synced 2024-11-24 02:06:07 +03:00
parent
9d90141c29
commit
073ca4267c
17 changed files with 188 additions and 77 deletions
|
@ -522,10 +522,7 @@ void Application::runExternalProgram(const QString &programTemplate, const BitTo
|
||||||
str.replace(i, 2, torrent->contentPath().toString());
|
str.replace(i, 2, torrent->contentPath().toString());
|
||||||
break;
|
break;
|
||||||
case u'G':
|
case u'G':
|
||||||
{
|
str.replace(i, 2, Utils::String::joinIntoString(torrent->tags(), u","_s));
|
||||||
const TagSet &tags = torrent->tags();
|
|
||||||
str.replace(i, 2, QStringList(tags.cbegin(), tags.cend()).join(u","_s));
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case u'I':
|
case u'I':
|
||||||
str.replace(i, 2, (torrent->infoHash().v1().isValid() ? torrent->infoHash().v1().toString() : u"-"_s));
|
str.replace(i, 2, (torrent->infoHash().v1().isValid() ? torrent->infoHash().v1().toString() : u"-"_s));
|
||||||
|
|
|
@ -42,6 +42,7 @@ add_library(qbt_base STATIC
|
||||||
bittorrent/torrentinfo.h
|
bittorrent/torrentinfo.h
|
||||||
bittorrent/tracker.h
|
bittorrent/tracker.h
|
||||||
bittorrent/trackerentry.h
|
bittorrent/trackerentry.h
|
||||||
|
concepts/explicitlyconvertibleto.h
|
||||||
concepts/stringable.h
|
concepts/stringable.h
|
||||||
digest32.h
|
digest32.h
|
||||||
exceptions.h
|
exceptions.h
|
||||||
|
|
|
@ -854,7 +854,7 @@ namespace
|
||||||
query.bindValue(DB_COLUMN_NAME.placeholder, m_resumeData.name);
|
query.bindValue(DB_COLUMN_NAME.placeholder, m_resumeData.name);
|
||||||
query.bindValue(DB_COLUMN_CATEGORY.placeholder, m_resumeData.category);
|
query.bindValue(DB_COLUMN_CATEGORY.placeholder, m_resumeData.category);
|
||||||
query.bindValue(DB_COLUMN_TAGS.placeholder, (m_resumeData.tags.isEmpty()
|
query.bindValue(DB_COLUMN_TAGS.placeholder, (m_resumeData.tags.isEmpty()
|
||||||
? QString() : QStringList(m_resumeData.tags.cbegin(), m_resumeData.tags.cend()).join(u","_s)));
|
? QString() : Utils::String::joinIntoString(m_resumeData.tags, u","_s)));
|
||||||
query.bindValue(DB_COLUMN_CONTENT_LAYOUT.placeholder, Utils::String::fromEnum(m_resumeData.contentLayout));
|
query.bindValue(DB_COLUMN_CONTENT_LAYOUT.placeholder, Utils::String::fromEnum(m_resumeData.contentLayout));
|
||||||
query.bindValue(DB_COLUMN_RATIO_LIMIT.placeholder, static_cast<int>(m_resumeData.ratioLimit * 1000));
|
query.bindValue(DB_COLUMN_RATIO_LIMIT.placeholder, static_cast<int>(m_resumeData.ratioLimit * 1000));
|
||||||
query.bindValue(DB_COLUMN_SEEDING_TIME_LIMIT.placeholder, m_resumeData.seedingTimeLimit);
|
query.bindValue(DB_COLUMN_SEEDING_TIME_LIMIT.placeholder, m_resumeData.seedingTimeLimit);
|
||||||
|
|
|
@ -552,8 +552,11 @@ SessionImpl::SessionImpl(QObject *parent)
|
||||||
}
|
}
|
||||||
|
|
||||||
const QStringList storedTags = m_storedTags.get();
|
const QStringList storedTags = m_storedTags.get();
|
||||||
m_tags.insert(storedTags.cbegin(), storedTags.cend());
|
for (const QString &tagStr : storedTags)
|
||||||
std::erase_if(m_tags, [](const Tag &tag) { return !tag.isValid(); });
|
{
|
||||||
|
if (const Tag tag {tagStr}; tag.isValid())
|
||||||
|
m_tags.insert(tag);
|
||||||
|
}
|
||||||
|
|
||||||
updateSeedingLimitTimer();
|
updateSeedingLimitTimer();
|
||||||
populateAdditionalTrackers();
|
populateAdditionalTrackers();
|
||||||
|
|
35
src/base/concepts/explicitlyconvertibleto.h
Normal file
35
src/base/concepts/explicitlyconvertibleto.h
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2023 Mike Tzou (Chocobo1)
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
template <typename From, typename To>
|
||||||
|
concept ExplicitlyConvertibleTo = requires (From f)
|
||||||
|
{
|
||||||
|
static_cast<To>(f);
|
||||||
|
};
|
|
@ -29,12 +29,9 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <concepts>
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
#include "algorithm.h"
|
|
||||||
|
|
||||||
template <typename T, typename Compare = std::less<T>>
|
template <typename T, typename Compare = std::less<T>>
|
||||||
class OrderedSet : public std::set<T, Compare>
|
class OrderedSet : public std::set<T, Compare>
|
||||||
{
|
{
|
||||||
|
@ -70,25 +67,6 @@ public:
|
||||||
return BaseType::empty();
|
return BaseType::empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
QString join(const QString &separator) const
|
|
||||||
requires std::same_as<value_type, QString>
|
|
||||||
{
|
|
||||||
auto iter = BaseType::cbegin();
|
|
||||||
if (iter == BaseType::cend())
|
|
||||||
return {};
|
|
||||||
|
|
||||||
QString ret = *iter;
|
|
||||||
++iter;
|
|
||||||
|
|
||||||
while (iter != BaseType::cend())
|
|
||||||
{
|
|
||||||
ret.push_back(separator + *iter);
|
|
||||||
++iter;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool remove(const key_type &value)
|
bool remove(const key_type &value)
|
||||||
{
|
{
|
||||||
return (BaseType::erase(value) > 0);
|
return (BaseType::erase(value) > 0);
|
||||||
|
|
|
@ -31,7 +31,6 @@
|
||||||
#include <QDataStream>
|
#include <QDataStream>
|
||||||
|
|
||||||
#include "base/concepts/stringable.h"
|
#include "base/concepts/stringable.h"
|
||||||
#include "base/global.h"
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
|
|
|
@ -119,14 +119,3 @@ std::optional<double> Utils::String::parseDouble(const QString &string)
|
||||||
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString Utils::String::join(const QList<QStringView> &strings, const QStringView separator)
|
|
||||||
{
|
|
||||||
if (strings.empty())
|
|
||||||
return {};
|
|
||||||
|
|
||||||
QString ret = strings[0].toString();
|
|
||||||
for (int i = 1; i < strings.count(); ++i)
|
|
||||||
ret += (separator + strings[i]);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
/*
|
/*
|
||||||
* Bittorrent Client using Qt and libtorrent.
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2023 Mike Tzou (Chocobo1)
|
||||||
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru>
|
* Copyright (C) 2015 Vladimir Golovnev <glassez@yandex.ru>
|
||||||
* Copyright (C) 2006 Christophe Dumez <chris@qbittorrent.org>
|
* Copyright (C) 2006 Christophe Dumez <chris@qbittorrent.org>
|
||||||
*
|
*
|
||||||
|
@ -29,14 +30,15 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <numeric>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
|
||||||
#include <QChar>
|
#include <QChar>
|
||||||
#include <QMetaEnum>
|
#include <QMetaEnum>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
#include <Qt>
|
|
||||||
#include <QtContainerFwd>
|
#include <QtContainerFwd>
|
||||||
|
|
||||||
|
#include "base/concepts/explicitlyconvertibleto.h"
|
||||||
#include "base/global.h"
|
#include "base/global.h"
|
||||||
|
|
||||||
namespace Utils::String
|
namespace Utils::String
|
||||||
|
@ -63,10 +65,37 @@ namespace Utils::String
|
||||||
|
|
||||||
QStringList splitCommand(const QString &command);
|
QStringList splitCommand(const QString &command);
|
||||||
|
|
||||||
QString join(const QList<QStringView> &strings, QStringView separator);
|
|
||||||
|
|
||||||
QString fromDouble(double n, int precision);
|
QString fromDouble(double n, int precision);
|
||||||
|
|
||||||
|
template <typename Container>
|
||||||
|
QString joinIntoString(const Container &container, const QString &separator)
|
||||||
|
requires ExplicitlyConvertibleTo<typename Container::value_type, QString>
|
||||||
|
{
|
||||||
|
auto iter = container.cbegin();
|
||||||
|
const auto end = container.cend();
|
||||||
|
if (iter == end)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
const qsizetype totalLength = std::accumulate(iter, end, (separator.size() * (container.size() - 1))
|
||||||
|
, [](const qsizetype total, const typename Container::value_type &value)
|
||||||
|
{
|
||||||
|
return total + QString(value).size();
|
||||||
|
});
|
||||||
|
|
||||||
|
QString ret;
|
||||||
|
ret.reserve(totalLength);
|
||||||
|
ret.append(QString(*iter));
|
||||||
|
++iter;
|
||||||
|
|
||||||
|
while (iter != end)
|
||||||
|
{
|
||||||
|
ret.append(separator + QString(*iter));
|
||||||
|
++iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QString fromEnum(const T &value)
|
QString fromEnum(const T &value)
|
||||||
requires std::is_enum_v<T>
|
requires std::is_enum_v<T>
|
||||||
|
|
|
@ -58,6 +58,7 @@
|
||||||
#include "base/utils/compare.h"
|
#include "base/utils/compare.h"
|
||||||
#include "base/utils/fs.h"
|
#include "base/utils/fs.h"
|
||||||
#include "base/utils/misc.h"
|
#include "base/utils/misc.h"
|
||||||
|
#include "base/utils/string.h"
|
||||||
#include "lineedit.h"
|
#include "lineedit.h"
|
||||||
#include "torrenttagsdialog.h"
|
#include "torrenttagsdialog.h"
|
||||||
|
|
||||||
|
@ -363,7 +364,7 @@ AddNewTorrentDialog::AddNewTorrentDialog(const BitTorrent::TorrentDescriptor &to
|
||||||
|
|
||||||
connect(m_ui->categoryComboBox, &QComboBox::currentIndexChanged, this, &AddNewTorrentDialog::categoryChanged);
|
connect(m_ui->categoryComboBox, &QComboBox::currentIndexChanged, this, &AddNewTorrentDialog::categoryChanged);
|
||||||
|
|
||||||
m_ui->tagsLineEdit->setText(QStringList(m_torrentParams.tags.cbegin(), m_torrentParams.tags.cend()).join(u", "_s));
|
m_ui->tagsLineEdit->setText(Utils::String::joinIntoString(m_torrentParams.tags, u", "_s));
|
||||||
connect(m_ui->tagsEditButton, &QAbstractButton::clicked, this, [this]
|
connect(m_ui->tagsEditButton, &QAbstractButton::clicked, this, [this]
|
||||||
{
|
{
|
||||||
auto *dlg = new TorrentTagsDialog(m_torrentParams.tags, this);
|
auto *dlg = new TorrentTagsDialog(m_torrentParams.tags, this);
|
||||||
|
@ -371,7 +372,7 @@ AddNewTorrentDialog::AddNewTorrentDialog(const BitTorrent::TorrentDescriptor &to
|
||||||
connect(dlg, &TorrentTagsDialog::accepted, this, [this, dlg]
|
connect(dlg, &TorrentTagsDialog::accepted, this, [this, dlg]
|
||||||
{
|
{
|
||||||
m_torrentParams.tags = dlg->tags();
|
m_torrentParams.tags = dlg->tags();
|
||||||
m_ui->tagsLineEdit->setText(QStringList(m_torrentParams.tags.cbegin(), m_torrentParams.tags.cend()).join(u", "_s));
|
m_ui->tagsLineEdit->setText(Utils::String::joinIntoString(m_torrentParams.tags, u", "_s));
|
||||||
});
|
});
|
||||||
dlg->open();
|
dlg->open();
|
||||||
});
|
});
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
#include "base/bittorrent/session.h"
|
#include "base/bittorrent/session.h"
|
||||||
#include "base/bittorrent/torrent.h"
|
#include "base/bittorrent/torrent.h"
|
||||||
#include "base/utils/compare.h"
|
#include "base/utils/compare.h"
|
||||||
|
#include "base/utils/string.h"
|
||||||
#include "flowlayout.h"
|
#include "flowlayout.h"
|
||||||
#include "fspathedit.h"
|
#include "fspathedit.h"
|
||||||
#include "torrenttagsdialog.h"
|
#include "torrenttagsdialog.h"
|
||||||
|
@ -112,7 +113,7 @@ AddTorrentParamsWidget::AddTorrentParamsWidget(BitTorrent::AddTorrentParams addT
|
||||||
connect(dlg, &TorrentTagsDialog::accepted, this, [this, dlg]
|
connect(dlg, &TorrentTagsDialog::accepted, this, [this, dlg]
|
||||||
{
|
{
|
||||||
m_addTorrentParams.tags = dlg->tags();
|
m_addTorrentParams.tags = dlg->tags();
|
||||||
m_ui->tagsLineEdit->setText(QStringList(m_addTorrentParams.tags.cbegin(), m_addTorrentParams.tags.cend()).join(u", "_s));
|
m_ui->tagsLineEdit->setText(Utils::String::joinIntoString(m_addTorrentParams.tags, u", "_s));
|
||||||
});
|
});
|
||||||
dlg->open();
|
dlg->open();
|
||||||
});
|
});
|
||||||
|
@ -230,7 +231,7 @@ void AddTorrentParamsWidget::populate()
|
||||||
m_addTorrentParams.stopCondition = data.value<BitTorrent::Torrent::StopCondition>();
|
m_addTorrentParams.stopCondition = data.value<BitTorrent::Torrent::StopCondition>();
|
||||||
});
|
});
|
||||||
|
|
||||||
m_ui->tagsLineEdit->setText(QStringList(m_addTorrentParams.tags.cbegin(), m_addTorrentParams.tags.cend()).join(u", "_s));
|
m_ui->tagsLineEdit->setText(Utils::String::joinIntoString(m_addTorrentParams.tags, u", "_s));
|
||||||
|
|
||||||
m_ui->startTorrentComboBox->disconnect(this);
|
m_ui->startTorrentComboBox->disconnect(this);
|
||||||
m_ui->startTorrentComboBox->setCurrentIndex(m_addTorrentParams.addPaused
|
m_ui->startTorrentComboBox->setCurrentIndex(m_addTorrentParams.addPaused
|
||||||
|
|
|
@ -380,10 +380,7 @@ QString TransferListModel::displayValue(const BitTorrent::Torrent *torrent, cons
|
||||||
case TR_CATEGORY:
|
case TR_CATEGORY:
|
||||||
return torrent->category();
|
return torrent->category();
|
||||||
case TR_TAGS:
|
case TR_TAGS:
|
||||||
{
|
return Utils::String::joinIntoString(torrent->tags(), u", "_s);
|
||||||
const TagSet &tags = torrent->tags();
|
|
||||||
return QStringList(tags.cbegin(), tags.cend()).join(u", "_s);
|
|
||||||
}
|
|
||||||
case TR_ADD_DATE:
|
case TR_ADD_DATE:
|
||||||
return QLocale().toString(torrent->addedTime().toLocalTime(), QLocale::ShortFormat);
|
return QLocale().toString(torrent->addedTime().toLocalTime(), QLocale::ShortFormat);
|
||||||
case TR_SEED_DATE:
|
case TR_SEED_DATE:
|
||||||
|
|
|
@ -29,7 +29,6 @@
|
||||||
#include "serialize_torrent.h"
|
#include "serialize_torrent.h"
|
||||||
|
|
||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QStringList>
|
|
||||||
#include <QVector>
|
#include <QVector>
|
||||||
|
|
||||||
#include "base/bittorrent/infohash.h"
|
#include "base/bittorrent/infohash.h"
|
||||||
|
@ -37,6 +36,7 @@
|
||||||
#include "base/bittorrent/trackerentry.h"
|
#include "base/bittorrent/trackerentry.h"
|
||||||
#include "base/path.h"
|
#include "base/path.h"
|
||||||
#include "base/tagset.h"
|
#include "base/tagset.h"
|
||||||
|
#include "base/utils/string.h"
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
|
@ -106,8 +106,6 @@ QVariantMap serialize(const BitTorrent::Torrent &torrent)
|
||||||
: (QDateTime::currentDateTime().toSecsSinceEpoch() - timeSinceActivity);
|
: (QDateTime::currentDateTime().toSecsSinceEpoch() - timeSinceActivity);
|
||||||
};
|
};
|
||||||
|
|
||||||
const TagSet &tags = torrent.tags();
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
{KEY_TORRENT_ID, torrent.id().toString()},
|
{KEY_TORRENT_ID, torrent.id().toString()},
|
||||||
{KEY_TORRENT_INFOHASHV1, torrent.infoHash().v1().toString()},
|
{KEY_TORRENT_INFOHASHV1, torrent.infoHash().v1().toString()},
|
||||||
|
@ -130,7 +128,7 @@ QVariantMap serialize(const BitTorrent::Torrent &torrent)
|
||||||
{KEY_TORRENT_FIRST_LAST_PIECE_PRIO, torrent.hasFirstLastPiecePriority()},
|
{KEY_TORRENT_FIRST_LAST_PIECE_PRIO, torrent.hasFirstLastPiecePriority()},
|
||||||
|
|
||||||
{KEY_TORRENT_CATEGORY, torrent.category()},
|
{KEY_TORRENT_CATEGORY, torrent.category()},
|
||||||
{KEY_TORRENT_TAGS, QStringList(tags.cbegin(), tags.cend()).join(u", "_s)},
|
{KEY_TORRENT_TAGS, Utils::String::joinIntoString(torrent.tags(), u", "_s)},
|
||||||
{KEY_TORRENT_SUPER_SEEDING, torrent.superSeeding()},
|
{KEY_TORRENT_SUPER_SEEDING, torrent.superSeeding()},
|
||||||
{KEY_TORRENT_FORCE_START, torrent.isForced()},
|
{KEY_TORRENT_FORCE_START, torrent.isForced()},
|
||||||
{KEY_TORRENT_SAVE_PATH, torrent.savePath().toString()},
|
{KEY_TORRENT_SAVE_PATH, torrent.savePath().toString()},
|
||||||
|
|
|
@ -8,6 +8,7 @@ include_directories("../src")
|
||||||
set(testFiles
|
set(testFiles
|
||||||
testalgorithm.cpp
|
testalgorithm.cpp
|
||||||
testbittorrenttrackerentry.cpp
|
testbittorrenttrackerentry.cpp
|
||||||
|
testconceptsexplicitlyconvertibleto.cpp
|
||||||
testconceptsstringable.cpp
|
testconceptsstringable.cpp
|
||||||
testglobal.cpp
|
testglobal.cpp
|
||||||
testorderedset.cpp
|
testorderedset.cpp
|
||||||
|
|
54
test/testconceptsexplicitlyconvertibleto.cpp
Normal file
54
test/testconceptsexplicitlyconvertibleto.cpp
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Bittorrent Client using Qt and libtorrent.
|
||||||
|
* Copyright (C) 2023 Mike Tzou (Chocobo1)
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QString>
|
||||||
|
#include <QTest>
|
||||||
|
|
||||||
|
#include "base/concepts/explicitlyconvertibleto.h"
|
||||||
|
|
||||||
|
class TestExplicitlyConvertibleTo final : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_DISABLE_COPY_MOVE(TestExplicitlyConvertibleTo)
|
||||||
|
|
||||||
|
public:
|
||||||
|
TestExplicitlyConvertibleTo() = default;
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void testExplicitlyConvertibleTo() const
|
||||||
|
{
|
||||||
|
static_assert(ExplicitlyConvertibleTo<const char *, std::string>);
|
||||||
|
static_assert(!ExplicitlyConvertibleTo<std::string, const char *>);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_APPLESS_MAIN(TestExplicitlyConvertibleTo)
|
||||||
|
#include "testconceptsexplicitlyconvertibleto.moc"
|
|
@ -33,6 +33,7 @@
|
||||||
|
|
||||||
#include "base/global.h"
|
#include "base/global.h"
|
||||||
#include "base/orderedset.h"
|
#include "base/orderedset.h"
|
||||||
|
#include "base/utils/string.h"
|
||||||
|
|
||||||
class TestOrderedSet final : public QObject
|
class TestOrderedSet final : public QObject
|
||||||
{
|
{
|
||||||
|
@ -57,7 +58,7 @@ private slots:
|
||||||
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
||||||
set.intersect({u"c"_s, u"a"_s});
|
set.intersect({u"c"_s, u"a"_s});
|
||||||
QCOMPARE(set.size(), 2);
|
QCOMPARE(set.size(), 2);
|
||||||
QCOMPARE(set.join(u","_s), u"a,c"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,c"_s);
|
||||||
|
|
||||||
OrderedSet<QString> emptySet;
|
OrderedSet<QString> emptySet;
|
||||||
emptySet.intersect({u"a"_s}).intersect({u"c"_s});;
|
emptySet.intersect({u"a"_s}).intersect({u"c"_s});;
|
||||||
|
@ -73,24 +74,15 @@ private slots:
|
||||||
QVERIFY(emptySet.isEmpty());
|
QVERIFY(emptySet.isEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
void testJoin() const
|
|
||||||
{
|
|
||||||
const OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
|
||||||
QCOMPARE(set.join(u","_s), u"a,b,c"_s);
|
|
||||||
|
|
||||||
const OrderedSet<QString> emptySet;
|
|
||||||
QCOMPARE(emptySet.join(u","_s), u""_s);
|
|
||||||
}
|
|
||||||
|
|
||||||
void testRemove() const
|
void testRemove() const
|
||||||
{
|
{
|
||||||
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
||||||
QVERIFY(!set.remove(u"z"_s));
|
QVERIFY(!set.remove(u"z"_s));
|
||||||
QCOMPARE(set.join(u","_s), u"a,b,c"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,b,c"_s);
|
||||||
QVERIFY(set.remove(u"b"_s));
|
QVERIFY(set.remove(u"b"_s));
|
||||||
QCOMPARE(set.join(u","_s), u"a,c"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,c"_s);
|
||||||
QVERIFY(set.remove(u"a"_s));
|
QVERIFY(set.remove(u"a"_s));
|
||||||
QCOMPARE(set.join(u","_s), u"c"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"c"_s);
|
||||||
QVERIFY(set.remove(u"c"_s));
|
QVERIFY(set.remove(u"c"_s));
|
||||||
QVERIFY(set.isEmpty());
|
QVERIFY(set.isEmpty());
|
||||||
|
|
||||||
|
@ -107,15 +99,15 @@ private slots:
|
||||||
|
|
||||||
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
||||||
set.unite(newData1);
|
set.unite(newData1);
|
||||||
QCOMPARE(set.join(u","_s), u"a,b,c,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,b,c,z"_s);
|
||||||
set.unite(newData2);
|
set.unite(newData2);
|
||||||
QCOMPARE(set.join(u","_s), u"a,b,c,y,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,b,c,y,z"_s);
|
||||||
set.unite(newData3);
|
set.unite(newData3);
|
||||||
QCOMPARE(set.join(u","_s), u"a,b,c,d,e,y,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(set, u","_s), u"a,b,c,d,e,y,z"_s);
|
||||||
|
|
||||||
OrderedSet<QString> emptySet;
|
OrderedSet<QString> emptySet;
|
||||||
emptySet.unite(newData1).unite(newData2).unite(newData3);
|
emptySet.unite(newData1).unite(newData2).unite(newData3);
|
||||||
QCOMPARE(emptySet.join(u","_s), u"c,d,e,y,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(emptySet, u","_s), u"c,d,e,y,z"_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
void testUnited() const
|
void testUnited() const
|
||||||
|
@ -126,11 +118,11 @@ private slots:
|
||||||
|
|
||||||
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
OrderedSet<QString> set {u"a"_s, u"b"_s, u"c"_s};
|
||||||
|
|
||||||
QCOMPARE(set.united(newData1).join(u","_s), u"a,b,c,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(set.united(newData1), u","_s), u"a,b,c,z"_s);
|
||||||
QCOMPARE(set.united(newData2).join(u","_s), u"a,b,c,y"_s);
|
QCOMPARE(Utils::String::joinIntoString(set.united(newData2), u","_s), u"a,b,c,y"_s);
|
||||||
QCOMPARE(set.united(newData3).join(u","_s), u"a,b,c,d,e"_s);
|
QCOMPARE(Utils::String::joinIntoString(set.united(newData3), u","_s), u"a,b,c,d,e"_s);
|
||||||
|
|
||||||
QCOMPARE(OrderedSet<QString>().united(newData1).united(newData2).united(newData3).join(u","_s), u"c,d,e,y,z"_s);
|
QCOMPARE(Utils::String::joinIntoString(OrderedSet<QString>().united(newData1).united(newData2).united(newData3), u","_s), u"c,d,e,y,z"_s);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -26,12 +26,33 @@
|
||||||
* exception statement from your version.
|
* exception statement from your version.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QList>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QTest>
|
#include <QTest>
|
||||||
|
|
||||||
#include "base/global.h"
|
#include "base/global.h"
|
||||||
#include "base/utils/string.h"
|
#include "base/utils/string.h"
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
class MyString
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
MyString(const QString &str)
|
||||||
|
: m_str {str}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
explicit operator QString() const
|
||||||
|
{
|
||||||
|
return m_str;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
QString m_str;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
class TestUtilsString final : public QObject
|
class TestUtilsString final : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
@ -41,6 +62,21 @@ public:
|
||||||
TestUtilsString() = default;
|
TestUtilsString() = default;
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
|
void testJoinIntoString() const
|
||||||
|
{
|
||||||
|
const QList<QString> list1;
|
||||||
|
QCOMPARE(Utils::String::joinIntoString(list1, u","_s), u""_s);
|
||||||
|
|
||||||
|
const QList<QString> list2 {u"a"_s};
|
||||||
|
QCOMPARE(Utils::String::joinIntoString(list2, u","_s), u"a"_s);
|
||||||
|
|
||||||
|
const QList<QString> list3 {u"a"_s, u"b"_s};
|
||||||
|
QCOMPARE(Utils::String::joinIntoString(list3, u" , "_s), u"a , b"_s);
|
||||||
|
|
||||||
|
const QList<MyString> list4 {u"a"_s, u"b"_s, u"cd"_s};
|
||||||
|
QCOMPARE(Utils::String::joinIntoString(list4, u"++"_s), u"a++b++cd"_s);
|
||||||
|
}
|
||||||
|
|
||||||
void testSplitCommand() const
|
void testSplitCommand() const
|
||||||
{
|
{
|
||||||
QCOMPARE(Utils::String::splitCommand({}), {});
|
QCOMPARE(Utils::String::splitCommand({}), {});
|
||||||
|
|
Loading…
Reference in a new issue