Skip to content
Snippets Groups Projects
vcsbaseclientsettings.cpp 10.8 KiB
Newer Older
cerf's avatar
cerf committed
/**************************************************************************
**
** Copyright (c) 2013 Brian McGillion and Hugues Delorme
hjk's avatar
hjk committed
** Contact: http://www.qt-project.org/legal
cerf's avatar
cerf committed
**
hjk's avatar
hjk committed
** This file is part of Qt Creator.
cerf's avatar
cerf committed
**
hjk's avatar
hjk committed
** 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
** a written agreement between you and Digia.  For licensing terms and
** conditions see http://qt.digia.com/licensing.  For further information
** use the contact form at http://qt.digia.com/contact-us.
cerf's avatar
cerf committed
**
** GNU Lesser General Public License Usage
hjk's avatar
hjk committed
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file.  Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Digia gives you certain additional
** rights.  These rights are described in the Digia Qt LGPL Exception
cerf's avatar
cerf committed
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
hjk's avatar
hjk committed
****************************************************************************/
cerf's avatar
cerf committed

#include "vcsbaseclientsettings.h"

#include <utils/environment.h>
#include <utils/hostosinfo.h>
#include <utils/qtcassert.h>
#include <QSettings>
#include <QVariant>
class SettingValue
{
public:
    union Composite
    {
        QString *strPtr; // Union can't store class objects ...
        int intValue;
        bool boolValue;
    };

    SettingValue() :
        m_type(QVariant::Invalid)
    {
    }

    explicit SettingValue(const QVariant &v) :
        m_type(v.type())
    {
        switch (v.type()) {
        case QVariant::UInt:
            m_type = QVariant::Int;
        case QVariant::Int:
            m_comp.intValue = v.toInt();
            break;
        case QVariant::Bool:
            m_comp.boolValue = v.toBool();
            break;
        case QVariant::String:
            m_comp.strPtr = new QString(v.toString());
            break;
        default:
            m_type = QVariant::Invalid;
            break;
        }
    }

    SettingValue(const SettingValue &other) :
        m_comp(other.m_comp),
        m_type(other.type())
    {
        copyInternalString(other);
    }

    ~SettingValue()
    {
        deleteInternalString();
    }

    SettingValue &operator=(const SettingValue &other)
    {
        if (this != &other) {
            deleteInternalString();
            m_type = other.type();
            m_comp = other.m_comp;
            copyInternalString(other);
        }
        return *this;
    }

    QString stringValue(const QString &defaultString = QString()) const
    {
        if (type() == QVariant::String && m_comp.strPtr != 0)
            return *(m_comp.strPtr);
        return defaultString;
    }

    QVariant::Type type() const
    {
        return m_type;
    }

    static bool isUsableVariantType(QVariant::Type varType)
    {
        return varType == QVariant::UInt || varType == QVariant::Int ||
                varType == QVariant::Bool || varType == QVariant::String;
    }

    Composite m_comp;

private:
    void deleteInternalString()
    {
        if (m_type == QVariant::String && m_comp.strPtr != 0) {
            delete m_comp.strPtr;
            m_comp.strPtr = 0;
        }
    }

    void copyInternalString(const SettingValue &other)
    {
        if (type() == QVariant::String) {
            const QString *otherString = other.m_comp.strPtr;
            m_comp.strPtr = new QString(otherString != 0 ? *otherString : QString());
        }
    }

    QVariant::Type m_type;
};

bool operator==(const SettingValue &lhs, const SettingValue &rhs)
{
    if (lhs.type() == rhs.type()) {
        switch (lhs.type()) {
        case QVariant::Int:
            return lhs.m_comp.intValue == rhs.m_comp.intValue;
        case QVariant::Bool:
            return lhs.m_comp.boolValue == rhs.m_comp.boolValue;
        case QVariant::String:
            return lhs.stringValue() == rhs.stringValue();
        default:
            return false;
        }
    }
    return false;
}

} // Anonymous namespace

hjk's avatar
hjk committed
namespace VcsBase {
hjk's avatar
hjk committed

hjk's avatar
hjk committed
class VcsBaseClientSettingsPrivate : public QSharedData
hjk's avatar
hjk committed
    VcsBaseClientSettingsPrivate() {}
hjk's avatar
hjk committed
    VcsBaseClientSettingsPrivate(const VcsBaseClientSettingsPrivate &other) :
        QSharedData(other),
        m_valueHash(other.m_valueHash),
        m_defaultValueHash(other.m_defaultValueHash),
        m_settingsGroup(other.m_settingsGroup),
        m_binaryFullPath(other.m_binaryFullPath)
    {
    }

    QHash<QString, SettingValue> m_valueHash;
    QVariantHash m_defaultValueHash;
    QString m_settingsGroup;
    mutable QString m_binaryFullPath;
hjk's avatar
hjk committed
    \class VcsBase::VcsBaseClientSettings
    \brief The VcsBaseClientSettings class contains settings used in
    VcsBaseClient.
hjk's avatar
hjk committed
    \sa VcsBase::VcsBaseClient
hjk's avatar
hjk committed
const QLatin1String VcsBaseClientSettings::binaryPathKey("BinaryPath");
const QLatin1String VcsBaseClientSettings::userNameKey("Username");
const QLatin1String VcsBaseClientSettings::userEmailKey("UserEmail");
const QLatin1String VcsBaseClientSettings::logCountKey("LogCount");
const QLatin1String VcsBaseClientSettings::promptOnSubmitKey("PromptOnSubmit");
const QLatin1String VcsBaseClientSettings::timeoutKey("Timeout");
const QLatin1String VcsBaseClientSettings::pathKey("Path");
hjk's avatar
hjk committed
VcsBaseClientSettings::VcsBaseClientSettings() :
    d(new Internal::VcsBaseClientSettingsPrivate)
cerf's avatar
cerf committed
{
    declareKey(binaryPathKey, QLatin1String(""));
    declareKey(userNameKey, QLatin1String(""));
    declareKey(userEmailKey, QLatin1String(""));
    declareKey(logCountKey, 100);
    declareKey(promptOnSubmitKey, true);
    declareKey(timeoutKey, 30);
    declareKey(pathKey, QString());
hjk's avatar
hjk committed
VcsBaseClientSettings::VcsBaseClientSettings(const VcsBaseClientSettings &other) :
hjk's avatar
hjk committed
VcsBaseClientSettings &VcsBaseClientSettings::operator=(const VcsBaseClientSettings &other)
cerf's avatar
cerf committed
{
    if (this != &other)
hjk's avatar
hjk committed
        d = other.d;
hjk's avatar
hjk committed
VcsBaseClientSettings::~VcsBaseClientSettings()
hjk's avatar
hjk committed
void VcsBaseClientSettings::writeSettings(QSettings *settings) const
cerf's avatar
cerf committed
{
    QTC_ASSERT(!settingsGroup().isEmpty(), return);

    settings->remove(settingsGroup());
    settings->beginGroup(settingsGroup());
    foreach (const QString &key, keys())
        settings->setValue(key, value(key));
    settings->endGroup();
hjk's avatar
hjk committed
void VcsBaseClientSettings::readSettings(const QSettings *settings)
cerf's avatar
cerf committed
{
    const QString keyRoot = settingsGroup() + QLatin1Char('/');
    foreach (const QString &key, keys()) {
        const QVariant value = settings->value(keyRoot + key, keyDefaultValue(key));
        // For some reason QSettings always return QVariant(QString) when the
        // key exists. The type is explicited to avoid wrong conversions
        switch (valueType(key)) {
        case QVariant::Int:
            setValue(key, value.toInt());
            break;
        case QVariant::Bool:
            setValue(key, value.toBool());
            break;
        case QVariant::String:
            setValue(key, value.toString());
            break;
        default:
            break;
        }
    }

    this->readLegacySettings(settings);
hjk's avatar
hjk committed
bool VcsBaseClientSettings::equals(const VcsBaseClientSettings &rhs) const
cerf's avatar
cerf committed
{
    if (this == &rhs)
        return true;
    return d->m_valueHash == rhs.d->m_valueHash;
hjk's avatar
hjk committed
QStringList VcsBaseClientSettings::keys() const
cerf's avatar
cerf committed
{
    return d->m_valueHash.keys();
hjk's avatar
hjk committed
bool VcsBaseClientSettings::hasKey(const QString &key) const
cerf's avatar
cerf committed
{
    return d->m_valueHash.contains(key);
hjk's avatar
hjk committed
int *VcsBaseClientSettings::intPointer(const QString &key)
cerf's avatar
cerf committed
{
    if (hasKey(key))
        return &(d->m_valueHash[key].m_comp.intValue);
    return 0;
hjk's avatar
hjk committed
bool *VcsBaseClientSettings::boolPointer(const QString &key)
cerf's avatar
cerf committed
{
    if (hasKey(key))
        return &(d->m_valueHash[key].m_comp.boolValue);
    return 0;
hjk's avatar
hjk committed
QString *VcsBaseClientSettings::stringPointer(const QString &key)
cerf's avatar
cerf committed
{
    if (hasKey(key) && valueType(key) == QVariant::String)
        return d->m_valueHash[key].m_comp.strPtr;
    return 0;
hjk's avatar
hjk committed
int VcsBaseClientSettings::intValue(const QString &key, int defaultValue) const
cerf's avatar
cerf committed
{
    if (hasKey(key) && valueType(key) == QVariant::Int)
        return d->m_valueHash[key].m_comp.intValue;
    return defaultValue;
hjk's avatar
hjk committed
bool VcsBaseClientSettings::boolValue(const QString &key, bool defaultValue) const
cerf's avatar
cerf committed
{
    if (hasKey(key) && valueType(key) == QVariant::Bool)
        return d->m_valueHash[key].m_comp.boolValue;
    return defaultValue;
hjk's avatar
hjk committed
QString VcsBaseClientSettings::stringValue(const QString &key, const QString &defaultValue) const
cerf's avatar
cerf committed
{
    if (hasKey(key))
        return d->m_valueHash[key].stringValue(defaultValue);
    return defaultValue;
hjk's avatar
hjk committed
QVariant VcsBaseClientSettings::value(const QString &key) const
cerf's avatar
cerf committed
{
    switch (valueType(key)) {
    case QVariant::Int:
        return intValue(key);
    case QVariant::Bool:
        return boolValue(key);
    case QVariant::String:
        return stringValue(key);
    case QVariant::Invalid:
        return QVariant();
    default:
        return QVariant();
    }
hjk's avatar
hjk committed
void VcsBaseClientSettings::setValue(const QString &key, const QVariant &v)
    if (SettingValue::isUsableVariantType(valueType(key))) {
        d->m_valueHash.insert(key, SettingValue(v));
        d->m_binaryFullPath.clear();
hjk's avatar
hjk committed
QVariant::Type VcsBaseClientSettings::valueType(const QString &key) const
    if (hasKey(key))
        return d->m_valueHash[key].type();
    return QVariant::Invalid;
QString VcsBaseClientSettings::binaryPath() const
{
    if (d->m_binaryFullPath.isEmpty()) {
        d->m_binaryFullPath = Utils::Environment::systemEnvironment().searchInPath(
                    stringValue(binaryPathKey), stringValue(pathKey).split(
                        Utils::HostOsInfo::pathListSeparator()));
    return d->m_binaryFullPath;
}

hjk's avatar
hjk committed
QString VcsBaseClientSettings::settingsGroup() const
cerf's avatar
cerf committed
{
    return d->m_settingsGroup;
hjk's avatar
hjk committed
void VcsBaseClientSettings::setSettingsGroup(const QString &group)
cerf's avatar
cerf committed
{
    d->m_settingsGroup = group;
cerf's avatar
cerf committed
}
hjk's avatar
hjk committed
void VcsBaseClientSettings::declareKey(const QString &key, const QVariant &defaultValue)
    if (SettingValue::isUsableVariantType(defaultValue.type())) {
        d->m_valueHash.insert(key, SettingValue(defaultValue));
        d->m_defaultValueHash.insert(key, defaultValue);
    }
hjk's avatar
hjk committed
QVariant VcsBaseClientSettings::keyDefaultValue(const QString &key) const
    if (d->m_defaultValueHash.contains(key))
        return d->m_defaultValueHash.value(key);
    return QVariant(valueType(key));
void VcsBaseClientSettings::readLegacySettings(const QSettings *settings)
{
    Q_UNUSED(settings);
}

hjk's avatar
hjk committed
} // namespace VcsBase