2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2017 The Qt Company Ltd.
|
2022-12-21 10:12:09 +01:00
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2016-02-02 18:26:51 +01:00
|
|
|
#pragma once
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2017-11-15 16:19:16 +01:00
|
|
|
#include "utils_global.h"
|
|
|
|
|
|
2022-05-24 12:31:20 +02:00
|
|
|
#include "filepath.h"
|
2013-03-25 17:13:18 +01:00
|
|
|
|
2014-05-12 15:19:13 +02:00
|
|
|
#include <QHash>
|
2014-03-03 18:57:42 +01:00
|
|
|
#include <QMessageBox>
|
2017-11-20 15:31:21 +01:00
|
|
|
#include <QVariantMap>
|
2013-03-25 17:13:18 +01:00
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
#include <memory>
|
2022-08-26 10:30:00 +02:00
|
|
|
#include <optional>
|
2017-12-11 11:30:01 +01:00
|
|
|
|
2017-11-15 16:19:16 +01:00
|
|
|
namespace Utils {
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2017-12-12 12:23:03 +01:00
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
// Helper:
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT int versionFromMap(const QVariantMap &data);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT int originalVersionFromMap(const QVariantMap &data);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QByteArray settingsIdFromMap(const QVariantMap &data);
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setVersionInMap(QVariantMap &data, int version);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setOriginalVersionInMap(QVariantMap &data, int version);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setSettingsIdInMap(QVariantMap &data, const QByteArray &id);
|
|
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
// Helpers:
|
|
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT int versionFromMap(const QVariantMap &data);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT int originalVersionFromMap(const QVariantMap &data);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QByteArray settingsIdFromMap(const QVariantMap &data);
|
|
|
|
|
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setVersionInMap(QVariantMap &data, int version);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setOriginalVersionInMap(QVariantMap &data, int version);
|
|
|
|
|
QTCREATOR_UTILS_EXPORT void setSettingsIdInMap(QVariantMap &data, const QByteArray &id);
|
|
|
|
|
|
2022-05-24 12:31:20 +02:00
|
|
|
class PersistentSettingsWriter;
|
2022-08-26 10:30:00 +02:00
|
|
|
using SettingsMergeResult = std::optional<QPair<QString, QVariant>>;
|
2017-12-12 17:38:23 +01:00
|
|
|
|
2017-11-20 15:31:21 +01:00
|
|
|
// --------------------------------------------------------------------
|
2017-12-14 15:23:17 +01:00
|
|
|
// SettingsAccessor:
|
2017-11-20 15:31:21 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
// Read/write files incl. error handling suitable for the UI:
|
2017-12-14 15:23:17 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT SettingsAccessor
|
2017-11-20 15:31:21 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2023-04-18 14:36:46 +02:00
|
|
|
SettingsAccessor();
|
2022-05-24 12:31:20 +02:00
|
|
|
virtual ~SettingsAccessor();
|
2017-12-11 11:30:01 +01:00
|
|
|
|
|
|
|
|
enum ProceedInfo { Continue, DiscardAndContinue };
|
2018-07-23 10:45:40 +02:00
|
|
|
using ButtonMap = QHash<QMessageBox::StandardButton, ProceedInfo>;
|
2017-12-11 11:30:01 +01:00
|
|
|
class Issue {
|
|
|
|
|
public:
|
2017-12-12 14:58:23 +01:00
|
|
|
enum class Type { ERROR, WARNING };
|
|
|
|
|
Issue(const QString &title, const QString &message, const Type type) :
|
|
|
|
|
title{title}, message{message}, type{type}
|
|
|
|
|
{ }
|
2017-12-11 11:30:01 +01:00
|
|
|
|
|
|
|
|
QMessageBox::StandardButtons allButtons() const;
|
|
|
|
|
|
|
|
|
|
QString title;
|
|
|
|
|
QString message;
|
2017-12-12 14:58:23 +01:00
|
|
|
Type type;
|
2017-12-11 11:30:01 +01:00
|
|
|
QMessageBox::StandardButton defaultButton = QMessageBox::NoButton;
|
|
|
|
|
QMessageBox::StandardButton escapeButton = QMessageBox::Ok;
|
|
|
|
|
QHash<QMessageBox::StandardButton, ProceedInfo> buttons = {{QMessageBox::Ok, ProceedInfo::Continue}};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class RestoreData {
|
|
|
|
|
public:
|
|
|
|
|
RestoreData() = default;
|
2019-05-28 13:49:26 +02:00
|
|
|
RestoreData(const FilePath &path, const QVariantMap &data) : path{path}, data{data} { }
|
2017-12-12 14:58:23 +01:00
|
|
|
RestoreData(const QString &title, const QString &message, const Issue::Type type) :
|
|
|
|
|
RestoreData(Issue(title, message, type))
|
|
|
|
|
{ }
|
2017-12-11 11:30:01 +01:00
|
|
|
RestoreData(const Issue &issue) : issue{issue} { }
|
2017-12-12 14:58:23 +01:00
|
|
|
|
|
|
|
|
bool hasIssue() const { return bool(issue); }
|
|
|
|
|
bool hasError() const { return hasIssue() && issue.value().type == Issue::Type::ERROR; }
|
|
|
|
|
bool hasWarning() const { return hasIssue() && issue.value().type == Issue::Type::WARNING; }
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
FilePath path;
|
2017-12-11 11:30:01 +01:00
|
|
|
QVariantMap data;
|
2022-08-26 10:30:00 +02:00
|
|
|
std::optional<Issue> issue;
|
2017-12-11 11:30:01 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
QVariantMap restoreSettings(QWidget *parent) const;
|
|
|
|
|
bool saveSettings(const QVariantMap &data, QWidget *parent) const;
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
void setBaseFilePath(const FilePath &baseFilePath) { m_baseFilePath = baseFilePath; }
|
2018-05-06 10:55:49 +03:00
|
|
|
void setReadOnly() { m_readOnly = true; }
|
2019-05-28 13:49:26 +02:00
|
|
|
FilePath baseFilePath() const { return m_baseFilePath; }
|
2017-11-20 15:31:21 +01:00
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
virtual RestoreData readData(const FilePath &path, QWidget *parent) const;
|
2022-08-26 10:30:00 +02:00
|
|
|
virtual std::optional<Issue> writeData(const FilePath &path,
|
|
|
|
|
const QVariantMap &data,
|
|
|
|
|
QWidget *parent) const;
|
2017-11-20 15:31:21 +01:00
|
|
|
|
2023-04-18 14:36:46 +02:00
|
|
|
void setDocType(const QString &docType) { m_docType = docType; }
|
|
|
|
|
void setApplicationDisplayName(const QString &name) { m_applicationDisplayName = name; }
|
|
|
|
|
|
2017-11-20 15:31:21 +01:00
|
|
|
protected:
|
2017-12-11 11:30:01 +01:00
|
|
|
// Report errors:
|
2019-05-28 13:49:26 +02:00
|
|
|
QVariantMap restoreSettings(const FilePath &settingsPath, QWidget *parent) const;
|
2020-11-21 01:04:56 +01:00
|
|
|
static ProceedInfo reportIssues(const Issue &issue, const FilePath &path, QWidget *parent);
|
2017-11-20 15:31:21 +01:00
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
virtual QVariantMap preprocessReadSettings(const QVariantMap &data) const;
|
|
|
|
|
virtual QVariantMap prepareToWriteSettings(const QVariantMap &data) const;
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
virtual RestoreData readFile(const FilePath &path) const;
|
2022-08-26 10:30:00 +02:00
|
|
|
virtual std::optional<Issue> writeFile(const FilePath &path, const QVariantMap &data) const;
|
2017-11-20 15:31:21 +01:00
|
|
|
|
2023-04-18 14:36:46 +02:00
|
|
|
QString m_docType;
|
|
|
|
|
QString m_applicationDisplayName;
|
|
|
|
|
|
2017-11-20 15:31:21 +01:00
|
|
|
private:
|
2019-05-28 13:49:26 +02:00
|
|
|
FilePath m_baseFilePath;
|
2017-11-20 15:31:21 +01:00
|
|
|
mutable std::unique_ptr<PersistentSettingsWriter> m_writer;
|
2018-05-06 10:55:49 +03:00
|
|
|
bool m_readOnly = false;
|
2017-11-20 15:31:21 +01:00
|
|
|
};
|
|
|
|
|
|
2017-12-12 12:23:03 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
// BackingUpSettingsAccessor:
|
|
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT BackUpStrategy
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
virtual ~BackUpStrategy() = default;
|
|
|
|
|
|
2019-12-17 14:07:53 +01:00
|
|
|
virtual FilePaths readFileCandidates(const FilePath &baseFileName) const;
|
2017-12-12 12:23:03 +01:00
|
|
|
// Return -1 if data1 is better that data2, 0 if both are equally worthwhile
|
|
|
|
|
// and 1 if data2 is better than data1
|
2017-12-14 15:23:17 +01:00
|
|
|
virtual int compare(const SettingsAccessor::RestoreData &data1,
|
|
|
|
|
const SettingsAccessor::RestoreData &data2) const;
|
2017-12-12 12:23:03 +01:00
|
|
|
|
2022-08-26 10:30:00 +02:00
|
|
|
virtual std::optional<FilePath> backupName(const QVariantMap &oldData,
|
|
|
|
|
const FilePath &path,
|
|
|
|
|
const QVariantMap &data) const;
|
2017-12-12 12:23:03 +01:00
|
|
|
};
|
|
|
|
|
|
2017-12-14 15:23:17 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT BackingUpSettingsAccessor : public SettingsAccessor
|
2017-12-12 12:23:03 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2023-04-18 14:36:46 +02:00
|
|
|
BackingUpSettingsAccessor();
|
2017-12-12 12:23:03 +01:00
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
RestoreData readData(const FilePath &path, QWidget *parent) const override;
|
2022-08-26 10:30:00 +02:00
|
|
|
std::optional<Issue> writeData(const FilePath &path,
|
|
|
|
|
const QVariantMap &data,
|
|
|
|
|
QWidget *parent) const override;
|
2017-12-12 12:23:03 +01:00
|
|
|
|
|
|
|
|
BackUpStrategy *strategy() const { return m_strategy.get(); }
|
2023-04-18 14:14:39 +02:00
|
|
|
void setStrategy(std::unique_ptr<BackUpStrategy> &&strategy);
|
2017-12-12 12:23:03 +01:00
|
|
|
|
|
|
|
|
private:
|
2019-12-17 14:07:53 +01:00
|
|
|
FilePaths readFileCandidates(const FilePath &path) const;
|
|
|
|
|
RestoreData bestReadFileData(const FilePaths &candidates, QWidget *parent) const;
|
2019-05-28 13:49:26 +02:00
|
|
|
void backupFile(const FilePath &path, const QVariantMap &data, QWidget *parent) const;
|
2017-12-12 12:23:03 +01:00
|
|
|
|
|
|
|
|
std::unique_ptr<BackUpStrategy> m_strategy;
|
|
|
|
|
};
|
|
|
|
|
|
2017-11-15 13:49:44 +01:00
|
|
|
// --------------------------------------------------------------------
|
2017-12-12 14:58:23 +01:00
|
|
|
// UpgradingSettingsAccessor:
|
2017-11-15 13:49:44 +01:00
|
|
|
// --------------------------------------------------------------------
|
2017-11-20 16:19:05 +01:00
|
|
|
|
2017-12-12 14:58:23 +01:00
|
|
|
class UpgradingSettingsAccessor;
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT VersionedBackUpStrategy : public BackUpStrategy
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
VersionedBackUpStrategy(const UpgradingSettingsAccessor *accessor);
|
|
|
|
|
|
|
|
|
|
// Return -1 if data1 is better that data2, 0 if both are equally worthwhile
|
|
|
|
|
// and 1 if data2 is better than data1
|
2017-12-14 15:23:17 +01:00
|
|
|
int compare(const SettingsAccessor::RestoreData &data1,
|
|
|
|
|
const SettingsAccessor::RestoreData &data2) const override;
|
2017-12-12 14:58:23 +01:00
|
|
|
|
2022-08-26 10:30:00 +02:00
|
|
|
std::optional<FilePath> backupName(const QVariantMap &oldData,
|
|
|
|
|
const FilePath &path,
|
|
|
|
|
const QVariantMap &data) const override;
|
2017-12-12 14:58:23 +01:00
|
|
|
|
|
|
|
|
const UpgradingSettingsAccessor *accessor() const { return m_accessor; }
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
const UpgradingSettingsAccessor *m_accessor = nullptr;
|
|
|
|
|
};
|
|
|
|
|
|
2017-11-20 16:19:05 +01:00
|
|
|
// Handles updating a QVariantMap from version() to version() + 1
|
2017-11-15 16:19:16 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT VersionUpgrader
|
2017-11-15 13:49:44 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2017-11-22 20:49:53 +01:00
|
|
|
VersionUpgrader(const int version, const QString &extension);
|
2017-11-15 13:49:44 +01:00
|
|
|
virtual ~VersionUpgrader() = default;
|
|
|
|
|
|
2017-11-22 20:49:53 +01:00
|
|
|
int version() const;
|
|
|
|
|
QString backupExtension() const;
|
2017-11-15 13:49:44 +01:00
|
|
|
|
|
|
|
|
virtual QVariantMap upgrade(const QVariantMap &data) = 0;
|
|
|
|
|
|
|
|
|
|
protected:
|
2022-09-30 15:57:27 +02:00
|
|
|
using Change = QPair<QLatin1String, QLatin1String>;
|
2017-11-15 13:49:44 +01:00
|
|
|
QVariantMap renameKeys(const QList<Change> &changes, QVariantMap map) const;
|
2017-11-22 20:49:53 +01:00
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
const int m_version;
|
|
|
|
|
const QString m_extension;
|
2017-11-15 13:49:44 +01:00
|
|
|
};
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
class MergingSettingsAccessor;
|
2014-02-27 12:47:13 +01:00
|
|
|
|
2017-12-12 14:58:23 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT UpgradingSettingsAccessor : public BackingUpSettingsAccessor
|
2010-01-19 16:59:18 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2023-04-18 14:36:46 +02:00
|
|
|
UpgradingSettingsAccessor();
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2014-02-28 18:32:09 +01:00
|
|
|
int currentVersion() const;
|
|
|
|
|
int firstSupportedVersion() const;
|
2017-12-12 14:58:23 +01:00
|
|
|
int lastSupportedVersion() const;
|
|
|
|
|
|
|
|
|
|
QByteArray settingsId() const { return m_id; }
|
|
|
|
|
|
|
|
|
|
bool isValidVersionAndId(const int version, const QByteArray &id) const;
|
|
|
|
|
VersionUpgrader *upgrader(const int version) const;
|
|
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
RestoreData readData(const FilePath &path, QWidget *parent) const override;
|
2017-12-12 14:58:23 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
protected:
|
2017-12-12 14:58:23 +01:00
|
|
|
QVariantMap prepareToWriteSettings(const QVariantMap &data) const override;
|
|
|
|
|
|
|
|
|
|
void setSettingsId(const QByteArray &id) { m_id = id; }
|
|
|
|
|
|
|
|
|
|
bool addVersionUpgrader(std::unique_ptr<VersionUpgrader> &&upgrader);
|
|
|
|
|
RestoreData upgradeSettings(const RestoreData &data, const int targetVersion) const;
|
|
|
|
|
RestoreData validateVersionRange(const RestoreData &data) const;
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QByteArray m_id;
|
|
|
|
|
std::vector<std::unique_ptr<VersionUpgrader>> m_upgraders;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// --------------------------------------------------------------------
|
2017-12-12 17:38:23 +01:00
|
|
|
// MergingSettingsAccessor:
|
2017-12-12 14:58:23 +01:00
|
|
|
// --------------------------------------------------------------------
|
2014-02-28 18:32:09 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT MergingSettingsAccessor : public UpgradingSettingsAccessor
|
2017-12-12 14:58:23 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2017-12-12 17:38:23 +01:00
|
|
|
struct SettingsMergeData {
|
|
|
|
|
QVariantMap main;
|
|
|
|
|
QVariantMap secondary;
|
|
|
|
|
QString key;
|
|
|
|
|
};
|
2014-02-28 16:13:34 +01:00
|
|
|
|
2023-04-18 14:36:46 +02:00
|
|
|
MergingSettingsAccessor();
|
2018-01-18 11:46:41 +01:00
|
|
|
|
2019-05-28 13:49:26 +02:00
|
|
|
RestoreData readData(const FilePath &path, QWidget *parent) const final;
|
2017-12-11 11:30:01 +01:00
|
|
|
|
2017-12-14 15:23:17 +01:00
|
|
|
void setSecondaryAccessor(std::unique_ptr<SettingsAccessor> &&secondary);
|
2014-03-03 18:57:42 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
protected:
|
2017-12-15 10:30:09 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
RestoreData mergeSettings(const RestoreData &main, const RestoreData &secondary) const;
|
|
|
|
|
|
|
|
|
|
virtual SettingsMergeResult merge(const SettingsMergeData &global,
|
|
|
|
|
const SettingsMergeData &local) const = 0;
|
2020-11-21 01:04:56 +01:00
|
|
|
static bool isHouseKeepingKey(const QString &key);
|
2013-04-04 13:30:16 +02:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
virtual QVariantMap postprocessMerge(const QVariantMap &main, const QVariantMap &secondary,
|
|
|
|
|
const QVariantMap &result) const;
|
2014-02-27 12:47:13 +01:00
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
private:
|
2017-12-14 15:23:17 +01:00
|
|
|
std::unique_ptr<SettingsAccessor> m_secondaryAccessor;
|
2010-01-19 16:59:18 +01:00
|
|
|
};
|
|
|
|
|
|
2017-12-12 17:38:23 +01:00
|
|
|
using SettingsMergeFunction = std::function<SettingsMergeResult(const MergingSettingsAccessor::SettingsMergeData &,
|
|
|
|
|
const MergingSettingsAccessor::SettingsMergeData &)>;
|
|
|
|
|
QTCREATOR_UTILS_EXPORT QVariant mergeQVariantMaps(const QVariantMap &mainTree, const QVariantMap &secondaryTree,
|
|
|
|
|
const SettingsMergeFunction &merge);
|
|
|
|
|
|
2017-11-15 16:19:16 +01:00
|
|
|
} // namespace Utils
|