mirror of
https://github.com/qbittorrent/qBittorrent.git
synced 2025-01-06 07:07:25 +03:00
d3e7e8a630
Remove redundant fragmentation of logic. PR #17354.
129 lines
4.4 KiB
C++
129 lines
4.4 KiB
C++
/*
|
|
* Bittorrent Client using Qt and libtorrent.
|
|
* Copyright (C) 2016 Vladimir Golovnev <glassez@yandex.ru>
|
|
* Copyright (C) 2014 sledgehammer999 <hammered999@gmail.com>
|
|
*
|
|
* 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
|
|
|
|
#include <type_traits>
|
|
|
|
#include <QObject>
|
|
#include <QReadWriteLock>
|
|
#include <QTimer>
|
|
#include <QVariant>
|
|
#include <QVariantHash>
|
|
|
|
#include "base/interfaces/istringable.h"
|
|
#include "utils/string.h"
|
|
|
|
template <typename T>
|
|
struct IsQFlags : std::false_type {};
|
|
|
|
template <typename T>
|
|
struct IsQFlags<QFlags<T>> : std::true_type {};
|
|
|
|
// There are 2 ways for class `T` provide serialization support into `SettingsStorage`:
|
|
// 1. If the `T` state is intended for users to edit (via a text editor), then
|
|
// implement `IStringable` interface
|
|
// 2. Otherwise, use `Q_DECLARE_METATYPE(T)` and let `QMetaType` handle the serialization
|
|
class SettingsStorage final : public QObject
|
|
{
|
|
Q_OBJECT
|
|
Q_DISABLE_COPY_MOVE(SettingsStorage)
|
|
|
|
SettingsStorage();
|
|
~SettingsStorage();
|
|
|
|
public:
|
|
static void initInstance();
|
|
static void freeInstance();
|
|
static SettingsStorage *instance();
|
|
|
|
template <typename T>
|
|
T loadValue(const QString &key, const T &defaultValue = {}) const
|
|
{
|
|
if constexpr (std::is_base_of_v<IStringable, T>)
|
|
{
|
|
const QString value = loadValue(key, defaultValue.toString());
|
|
return T {value};
|
|
}
|
|
else if constexpr (std::is_enum_v<T>)
|
|
{
|
|
const auto value = loadValue<QString>(key);
|
|
return Utils::String::toEnum(value, defaultValue);
|
|
}
|
|
else if constexpr (IsQFlags<T>::value)
|
|
{
|
|
const typename T::Int value = loadValue(key, static_cast<typename T::Int>(defaultValue));
|
|
return T {value};
|
|
}
|
|
else if constexpr (std::is_same_v<T, QVariant>)
|
|
{
|
|
// fast path for loading QVariant
|
|
return loadValueImpl(key, defaultValue);
|
|
}
|
|
else
|
|
{
|
|
const QVariant value = loadValueImpl(key);
|
|
// check if retrieved value is convertible to T
|
|
return value.template canConvert<T>() ? value.template value<T>() : defaultValue;
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
void storeValue(const QString &key, const T &value)
|
|
{
|
|
if constexpr (std::is_base_of_v<IStringable, T>)
|
|
storeValueImpl(key, value.toString());
|
|
else if constexpr (std::is_enum_v<T>)
|
|
storeValueImpl(key, Utils::String::fromEnum(value));
|
|
else if constexpr (IsQFlags<T>::value)
|
|
storeValueImpl(key, static_cast<typename T::Int>(value));
|
|
else
|
|
storeValueImpl(key, QVariant::fromValue(value));
|
|
}
|
|
|
|
void removeValue(const QString &key);
|
|
bool hasKey(const QString &key) const;
|
|
|
|
public slots:
|
|
bool save();
|
|
|
|
private:
|
|
QVariant loadValueImpl(const QString &key, const QVariant &defaultValue = {}) const;
|
|
void storeValueImpl(const QString &key, const QVariant &value);
|
|
void readNativeSettings();
|
|
bool writeNativeSettings() const;
|
|
|
|
static SettingsStorage *m_instance;
|
|
|
|
const QString m_nativeSettingsName;
|
|
bool m_dirty = false;
|
|
QVariantHash m_data;
|
|
QTimer m_timer;
|
|
mutable QReadWriteLock m_lock;
|
|
};
|