2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2010-01-19 16:59:18 +01:00
|
|
|
**
|
2017-11-15 16:19:16 +01:00
|
|
|
** Copyright (C) 2017 The Qt Company Ltd.
|
2016-01-15 14:57:40 +01:00
|
|
|
** Contact: https://www.qt.io/licensing/
|
2010-01-19 16:59:18 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2010-01-19 16:59:18 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2016-01-15 14:57:40 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2010-01-19 16:59:18 +01:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2010-12-17 16:01:08 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
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"
|
|
|
|
|
|
|
|
|
|
#include "fileutils.h"
|
2017-11-17 21:53:54 +01:00
|
|
|
#include "optional.h"
|
2017-11-20 15:31:21 +01:00
|
|
|
#include "persistentsettings.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>
|
|
|
|
|
|
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-11-20 15:31:21 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
// BasicSettingsAccessor:
|
|
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
// Read/write files incl. error handling suitable for the UI:
|
2017-11-20 15:31:21 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT BasicSettingsAccessor
|
|
|
|
|
{
|
|
|
|
|
public:
|
2017-12-11 11:30:01 +01:00
|
|
|
BasicSettingsAccessor(const QString &docType, const QString &displayName,
|
|
|
|
|
const QString &applicationDisplayName);
|
|
|
|
|
virtual ~BasicSettingsAccessor() = default;
|
|
|
|
|
|
|
|
|
|
enum ProceedInfo { Continue, DiscardAndContinue };
|
|
|
|
|
typedef QHash<QMessageBox::StandardButton, ProceedInfo> ButtonMap;
|
|
|
|
|
class Issue {
|
|
|
|
|
public:
|
|
|
|
|
Issue(const QString &title, const QString &message) : title{title}, message{message} { }
|
|
|
|
|
|
|
|
|
|
QMessageBox::StandardButtons allButtons() const;
|
|
|
|
|
|
|
|
|
|
QString title;
|
|
|
|
|
QString message;
|
|
|
|
|
QMessageBox::StandardButton defaultButton = QMessageBox::NoButton;
|
|
|
|
|
QMessageBox::StandardButton escapeButton = QMessageBox::Ok;
|
|
|
|
|
QHash<QMessageBox::StandardButton, ProceedInfo> buttons = {{QMessageBox::Ok, ProceedInfo::Continue}};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class RestoreData {
|
|
|
|
|
public:
|
|
|
|
|
RestoreData() = default;
|
|
|
|
|
RestoreData(const Utils::FileName &path, const QVariantMap &data) : path{path}, data{data} { }
|
|
|
|
|
RestoreData(const QString &title, const QString &message) : RestoreData(Issue(title, message)) { }
|
|
|
|
|
RestoreData(const Issue &issue) : issue{issue} { }
|
|
|
|
|
Utils::FileName path;
|
|
|
|
|
QVariantMap data;
|
|
|
|
|
Utils::optional<Issue> issue;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
QVariantMap restoreSettings(QWidget *parent) const;
|
|
|
|
|
bool saveSettings(const QVariantMap &data, QWidget *parent) const;
|
|
|
|
|
|
|
|
|
|
const QString docType;
|
|
|
|
|
const QString displayName;
|
|
|
|
|
const QString applicationDisplayName;
|
|
|
|
|
|
|
|
|
|
void setBaseFilePath(const Utils::FileName &baseFilePath) { m_baseFilePath = baseFilePath; }
|
|
|
|
|
Utils::FileName baseFilePath() const { return m_baseFilePath; }
|
2017-11-20 15:31:21 +01:00
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
virtual RestoreData readData(const Utils::FileName &path, QWidget *parent) const;
|
|
|
|
|
virtual Utils::optional<Issue> writeData(const Utils::FileName &path, const QVariantMap &data) const;
|
2017-11-20 15:31:21 +01:00
|
|
|
|
|
|
|
|
protected:
|
2017-12-11 11:30:01 +01:00
|
|
|
// Report errors:
|
|
|
|
|
ProceedInfo reportIssues(const Issue &issue, const FileName &path, QWidget *parent) const;
|
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;
|
|
|
|
|
|
|
|
|
|
RestoreData readFile(const Utils::FileName &path) const;
|
|
|
|
|
Utils::optional<Issue> writeFile(const Utils::FileName &path, const QVariantMap &data) const;
|
2017-11-20 15:31:21 +01:00
|
|
|
|
|
|
|
|
private:
|
2017-12-11 11:30:01 +01:00
|
|
|
Utils::FileName m_baseFilePath;
|
2017-11-20 15:31:21 +01:00
|
|
|
mutable std::unique_ptr<PersistentSettingsWriter> m_writer;
|
|
|
|
|
};
|
|
|
|
|
|
2017-12-12 12:23:03 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
// BackingUpSettingsAccessor:
|
|
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT BackUpStrategy
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
virtual ~BackUpStrategy() = default;
|
|
|
|
|
|
|
|
|
|
virtual FileNameList readFileCandidates(const Utils::FileName &baseFileName) const;
|
|
|
|
|
// Return -1 if data1 is better that data2, 0 if both are equally worthwhile
|
|
|
|
|
// and 1 if data2 is better than data1
|
|
|
|
|
virtual int compare(const BasicSettingsAccessor::RestoreData &data1,
|
|
|
|
|
const BasicSettingsAccessor::RestoreData &data2) const;
|
|
|
|
|
|
|
|
|
|
virtual QString backupName(const QVariantMap &oldData,
|
|
|
|
|
const FileName &path, const QVariantMap &data) const;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT NoBackUpStrategy : public BackUpStrategy
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
FileNameList readFileCandidates(const Utils::FileName &baseFileName) const { return {baseFileName}; }
|
|
|
|
|
QString backupName(const QVariantMap &,
|
|
|
|
|
const FileName &path, const QVariantMap &) const final { return path.toString(); }
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT VersionedBackUpStrategy : public BackUpStrategy
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
// Return -1 if data1 is better that data2, 0 if both are equally worthwhile
|
|
|
|
|
// and 1 if data2 is better than data1
|
|
|
|
|
int compare(const BasicSettingsAccessor::RestoreData &data1,
|
|
|
|
|
const BasicSettingsAccessor::RestoreData &data2) const override;
|
|
|
|
|
|
|
|
|
|
QString backupName(const QVariantMap &oldData,
|
|
|
|
|
const FileName &path, const QVariantMap &data) const override;
|
|
|
|
|
|
|
|
|
|
bool isValidVersionAndId(const int version, const QByteArray &id) const;
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
virtual QByteArray id() const = 0;
|
|
|
|
|
virtual int firstSupportedVersion() const = 0;
|
|
|
|
|
virtual int currentVersion() const = 0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class QTCREATOR_UTILS_EXPORT BackingUpSettingsAccessor : public BasicSettingsAccessor
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
BackingUpSettingsAccessor(const QString &docType, const QString &displayName,
|
|
|
|
|
const QString &applicationDisplayName);
|
|
|
|
|
BackingUpSettingsAccessor(std::unique_ptr<BackUpStrategy> &&strategy,
|
|
|
|
|
const QString &docType, const QString &displayName,
|
|
|
|
|
const QString &applicationDisplayName);
|
|
|
|
|
|
|
|
|
|
RestoreData readData(const Utils::FileName &path, QWidget *parent) const;
|
|
|
|
|
Utils::optional<Issue> writeData(const Utils::FileName &path, const QVariantMap &data) const;
|
|
|
|
|
|
|
|
|
|
BackUpStrategy *strategy() const { return m_strategy.get(); }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
Utils::FileNameList readFileCandidates(const FileName &path) const;
|
|
|
|
|
RestoreData bestReadFileData(const FileNameList &candidates, QWidget *parent) const;
|
|
|
|
|
void backupFile(const FileName &path, const QVariantMap &data) const;
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<BackUpStrategy> m_strategy;
|
|
|
|
|
};
|
|
|
|
|
|
2017-11-15 13:49:44 +01:00
|
|
|
// --------------------------------------------------------------------
|
|
|
|
|
// VersionUpgrader:
|
|
|
|
|
// --------------------------------------------------------------------
|
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:
|
|
|
|
|
typedef QPair<QLatin1String,QLatin1String> Change;
|
|
|
|
|
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
|
|
|
|
2014-02-27 12:47:13 +01:00
|
|
|
class SettingsAccessorPrivate;
|
|
|
|
|
|
2017-12-12 12:23:03 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT SettingsAccessor : public BackingUpSettingsAccessor
|
2010-01-19 16:59:18 +01:00
|
|
|
{
|
|
|
|
|
public:
|
2017-12-12 12:23:03 +01:00
|
|
|
explicit SettingsAccessor(std::unique_ptr<BackUpStrategy> &&strategy,
|
|
|
|
|
const Utils::FileName &baseFile, const QString &docType,
|
2017-12-11 11:30:01 +01:00
|
|
|
const QString &displayName, const QString &appDisplayName);
|
2017-11-20 15:31:21 +01:00
|
|
|
~SettingsAccessor() override;
|
2010-01-19 16:59:18 +01:00
|
|
|
|
2014-02-28 18:32:09 +01:00
|
|
|
int currentVersion() const;
|
|
|
|
|
int firstSupportedVersion() const;
|
|
|
|
|
|
2017-11-15 15:12:36 +01:00
|
|
|
bool addVersionUpgrader(std::unique_ptr<VersionUpgrader> upgrader);
|
2014-02-28 16:13:34 +01:00
|
|
|
|
2018-01-18 11:46:41 +01:00
|
|
|
Utils::FileName projectUserFile() const;
|
|
|
|
|
Utils::FileName externalUserFile() const;
|
|
|
|
|
Utils::FileName sharedFile() const;
|
|
|
|
|
|
2017-12-12 12:23:03 +01:00
|
|
|
QByteArray settingsId() const;
|
|
|
|
|
|
2014-02-28 16:13:34 +01:00
|
|
|
protected:
|
2017-12-11 11:30:01 +01:00
|
|
|
RestoreData readData(const Utils::FileName &path, QWidget *parent) const final;
|
|
|
|
|
|
2017-11-15 12:41:05 +01:00
|
|
|
void setSettingsId(const QByteArray &id);
|
2014-09-02 06:26:13 +03:00
|
|
|
QVariantMap upgradeSettings(const QVariantMap &data) const;
|
2017-11-17 21:22:41 +01:00
|
|
|
QVariantMap upgradeSettings(const QVariantMap &data, const int targetVersion) const;
|
2014-03-03 18:57:42 +01:00
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
QVariantMap prepareToWriteSettings(const QVariantMap &data) const override;
|
2014-02-28 16:13:34 +01:00
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
virtual void storeSharedSettings(const QVariantMap &data) const;
|
2014-03-03 18:57:42 +01:00
|
|
|
|
2017-12-15 10:30:09 +01:00
|
|
|
QVariantMap mergeSettings(const QVariantMap &userMap, const QVariantMap &sharedMap) const;
|
|
|
|
|
|
2017-12-11 11:30:01 +01:00
|
|
|
Utils::optional<Issue> findIssues(const QVariantMap &data, const Utils::FileName &path) const;
|
2017-11-15 13:21:32 +01:00
|
|
|
|
2010-01-19 16:59:18 +01:00
|
|
|
private:
|
2017-12-11 11:30:01 +01:00
|
|
|
RestoreData readSharedSettings(QWidget *parent) const;
|
2013-04-04 13:30:16 +02:00
|
|
|
|
2014-03-03 18:57:42 +01:00
|
|
|
static QString differentEnvironmentMsg(const QString &projectName);
|
2014-02-27 17:19:22 +01:00
|
|
|
|
2014-02-27 12:47:13 +01:00
|
|
|
SettingsAccessorPrivate *d;
|
|
|
|
|
|
|
|
|
|
friend class SettingsAccessorPrivate;
|
2010-01-19 16:59:18 +01:00
|
|
|
};
|
|
|
|
|
|
2017-11-15 16:19:16 +01:00
|
|
|
} // namespace Utils
|