/* * Bittorrent Client using Qt and libtorrent. * Copyright (C) 2016 Vladimir Golovnev * Copyright (C) 2014 sledgehammer999 * * 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 #include #include #include #include #include #include "base/interfaces/istringable.h" #include "utils/string.h" template struct IsQFlags : std::false_type {}; template struct IsQFlags> : 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 T loadValue(const QString &key, const T &defaultValue = {}) const { if constexpr (std::is_base_of_v) { const QString value = loadValue(key, defaultValue.toString()); return T {value}; } else if constexpr (std::is_enum_v) { const auto value = loadValue(key); return Utils::String::toEnum(value, defaultValue); } else if constexpr (IsQFlags::value) { const typename T::Int value = loadValue(key, static_cast(defaultValue)); return T {value}; } else if constexpr (std::is_same_v) { // 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() ? value.template value() : defaultValue; } } template void storeValue(const QString &key, const T &value) { if constexpr (std::is_base_of_v) storeValueImpl(key, value.toString()); else if constexpr (std::is_enum_v) storeValueImpl(key, Utils::String::fromEnum(value)); else if constexpr (IsQFlags::value) storeValueImpl(key, static_cast(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; };