forked from qt-creator/qt-creator
Change-Id: I45778c9562ba530a36ddaf201f0c61d380d701cf Reviewed-by: Leena Miettinen <riitta-leena.miettinen@theqtcompany.com>
687 lines
19 KiB
C++
687 lines
19 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
|
** Contact: http://www.qt-project.org/legal
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** 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://www.qt.io/licensing. For further information
|
|
** use the contact form at http://www.qt.io/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
** following information to ensure the GNU Lesser General Public License
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
** 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
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "kit.h"
|
|
|
|
#include "kitmanager.h"
|
|
#include "ioutputparser.h"
|
|
#include "osparser.h"
|
|
#include "projectexplorerconstants.h"
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/fileutils.h>
|
|
#include <utils/macroexpander.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QApplication>
|
|
#include <QFileInfo>
|
|
#include <QIcon>
|
|
#include <QStyle>
|
|
#include <QTextStream>
|
|
#include <QUuid>
|
|
|
|
using namespace Core;
|
|
using namespace Utils;
|
|
|
|
const char ID_KEY[] = "PE.Profile.Id";
|
|
const char DISPLAYNAME_KEY[] = "PE.Profile.Name";
|
|
const char FILESYSTEMFRIENDLYNAME_KEY[] = "PE.Profile.FileSystemFriendlyName";
|
|
const char AUTODETECTED_KEY[] = "PE.Profile.AutoDetected";
|
|
const char AUTODETECTIONSOURCE_KEY[] = "PE.Profile.AutoDetectionSource";
|
|
const char SDK_PROVIDED_KEY[] = "PE.Profile.SDK";
|
|
const char DATA_KEY[] = "PE.Profile.Data";
|
|
const char ICON_KEY[] = "PE.Profile.Icon";
|
|
const char MUTABLE_INFO_KEY[] = "PE.Profile.MutableInfo";
|
|
const char STICKY_INFO_KEY[] = "PE.Profile.StickyInfo";
|
|
|
|
namespace ProjectExplorer {
|
|
namespace Internal {
|
|
|
|
// -------------------------------------------------------------------------
|
|
// KitPrivate
|
|
// -------------------------------------------------------------------------
|
|
|
|
|
|
class KitPrivate
|
|
{
|
|
Q_DECLARE_TR_FUNCTIONS(ProjectExplorer::Kit)
|
|
|
|
public:
|
|
KitPrivate(Id id, Kit *kit) :
|
|
m_id(id),
|
|
m_nestedBlockingLevel(0),
|
|
m_autodetected(false),
|
|
m_sdkProvided(false),
|
|
m_isValid(true),
|
|
m_hasWarning(false),
|
|
m_hasValidityInfo(false),
|
|
m_mustNotify(false)
|
|
{
|
|
if (!id.isValid())
|
|
m_id = Id::fromString(QUuid::createUuid().toString());
|
|
|
|
m_unexpandedDisplayName = QCoreApplication::translate("ProjectExplorer::Kit", "Unnamed");
|
|
m_iconPath = FileName::fromLatin1(":///DESKTOP///");
|
|
|
|
m_macroExpander.setDisplayName(tr("Kit"));
|
|
m_macroExpander.setAccumulating(true);
|
|
m_macroExpander.registerVariable("Kit:Id", tr("Kit ID"),
|
|
[kit] { return kit->id().toString(); });
|
|
m_macroExpander.registerVariable("Kit:FileSystemName", tr("Kit filesystem-friendly name"),
|
|
[kit] { return kit->fileSystemFriendlyName(); });
|
|
foreach (KitInformation *ki, KitManager::kitInformation())
|
|
ki->addToMacroExpander(kit, &m_macroExpander);
|
|
|
|
// This provides the same global fall back as the global expander
|
|
// without relying on the currentKit() discovery process there.
|
|
m_macroExpander.registerVariable(Constants::VAR_CURRENTKIT_NAME,
|
|
tr("The name of the currently active kit."),
|
|
[kit] { return kit->displayName(); },
|
|
false);
|
|
m_macroExpander.registerVariable(Constants::VAR_CURRENTKIT_FILESYSTEMNAME,
|
|
tr("The name of the currently active kit in a filesystem-friendly version."),
|
|
[kit] { return kit->fileSystemFriendlyName(); },
|
|
false);
|
|
m_macroExpander.registerVariable(Constants::VAR_CURRENTKIT_ID,
|
|
tr("The id of the currently active kit."),
|
|
[kit] { return kit->id().toString(); },
|
|
false);
|
|
}
|
|
|
|
QString m_unexpandedDisplayName;
|
|
QString m_fileSystemFriendlyName;
|
|
QString m_autoDetectionSource;
|
|
Id m_id;
|
|
int m_nestedBlockingLevel;
|
|
bool m_autodetected;
|
|
bool m_sdkProvided;
|
|
bool m_isValid;
|
|
bool m_hasWarning;
|
|
bool m_hasValidityInfo;
|
|
bool m_mustNotify;
|
|
QIcon m_icon;
|
|
FileName m_iconPath;
|
|
|
|
QHash<Core::Id, QVariant> m_data;
|
|
QSet<Core::Id> m_sticky;
|
|
QSet<Core::Id> m_mutable;
|
|
MacroExpander m_macroExpander;
|
|
};
|
|
|
|
} // namespace Internal
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Kit:
|
|
// -------------------------------------------------------------------------
|
|
|
|
Kit::Kit(Core::Id id) :
|
|
d(new Internal::KitPrivate(id, this))
|
|
{
|
|
foreach (KitInformation *sti, KitManager::kitInformation())
|
|
d->m_data.insert(sti->id(), sti->defaultValue(this));
|
|
|
|
d->m_icon = icon(d->m_iconPath);
|
|
}
|
|
|
|
Kit::Kit(const QVariantMap &data) :
|
|
d(new Internal::KitPrivate(Core::Id(), this))
|
|
{
|
|
d->m_id = Id::fromSetting(data.value(QLatin1String(ID_KEY)));
|
|
|
|
d->m_autodetected = data.value(QLatin1String(AUTODETECTED_KEY)).toBool();
|
|
d->m_autoDetectionSource = data.value(QLatin1String(AUTODETECTIONSOURCE_KEY)).toString();
|
|
|
|
// if we don't have that setting assume that autodetected implies sdk
|
|
QVariant value = data.value(QLatin1String(SDK_PROVIDED_KEY));
|
|
if (value.isValid())
|
|
d->m_sdkProvided = value.toBool();
|
|
else
|
|
d->m_sdkProvided = d->m_autodetected;
|
|
|
|
d->m_unexpandedDisplayName = data.value(QLatin1String(DISPLAYNAME_KEY),
|
|
d->m_unexpandedDisplayName).toString();
|
|
d->m_fileSystemFriendlyName = data.value(QLatin1String(FILESYSTEMFRIENDLYNAME_KEY)).toString();
|
|
d->m_iconPath = FileName::fromString(data.value(QLatin1String(ICON_KEY),
|
|
d->m_iconPath.toString()).toString());
|
|
d->m_icon = icon(d->m_iconPath);
|
|
|
|
QVariantMap extra = data.value(QLatin1String(DATA_KEY)).toMap();
|
|
d->m_data.clear(); // remove default values
|
|
const QVariantMap::ConstIterator cend = extra.constEnd();
|
|
for (QVariantMap::ConstIterator it = extra.constBegin(); it != cend; ++it)
|
|
d->m_data.insert(Id::fromString(it.key()), it.value());
|
|
|
|
QStringList mutableInfoList = data.value(QLatin1String(MUTABLE_INFO_KEY)).toStringList();
|
|
foreach (const QString &mutableInfo, mutableInfoList)
|
|
d->m_mutable.insert(Core::Id::fromString(mutableInfo));
|
|
|
|
QStringList stickyInfoList = data.value(QLatin1String(STICKY_INFO_KEY)).toStringList();
|
|
foreach (const QString &stickyInfo, stickyInfoList)
|
|
d->m_sticky.insert(Core::Id::fromString(stickyInfo));
|
|
}
|
|
|
|
Kit::~Kit()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
void Kit::blockNotification()
|
|
{
|
|
++d->m_nestedBlockingLevel;
|
|
}
|
|
|
|
void Kit::unblockNotification()
|
|
{
|
|
--d->m_nestedBlockingLevel;
|
|
if (d->m_nestedBlockingLevel > 0)
|
|
return;
|
|
if (d->m_mustNotify)
|
|
kitUpdated();
|
|
}
|
|
|
|
Kit *Kit::clone(bool keepName) const
|
|
{
|
|
Kit *k = new Kit;
|
|
if (keepName)
|
|
k->d->m_unexpandedDisplayName = d->m_unexpandedDisplayName;
|
|
else
|
|
k->d->m_unexpandedDisplayName = QCoreApplication::translate("ProjectExplorer::Kit", "Clone of %1")
|
|
.arg(d->m_unexpandedDisplayName);
|
|
k->d->m_autodetected = false;
|
|
k->d->m_data = d->m_data;
|
|
// Do not clone m_fileSystemFriendlyName, needs to be unique
|
|
k->d->m_isValid = d->m_isValid;
|
|
k->d->m_icon = d->m_icon;
|
|
k->d->m_iconPath = d->m_iconPath;
|
|
k->d->m_sticky = d->m_sticky;
|
|
k->d->m_mutable = d->m_mutable;
|
|
return k;
|
|
}
|
|
|
|
void Kit::copyFrom(const Kit *k)
|
|
{
|
|
KitGuard g(this);
|
|
d->m_data = k->d->m_data;
|
|
d->m_iconPath = k->d->m_iconPath;
|
|
d->m_icon = k->d->m_icon;
|
|
d->m_autodetected = k->d->m_autodetected;
|
|
d->m_autoDetectionSource = k->d->m_autoDetectionSource;
|
|
d->m_unexpandedDisplayName = k->d->m_unexpandedDisplayName;
|
|
d->m_fileSystemFriendlyName = k->d->m_fileSystemFriendlyName;
|
|
d->m_mustNotify = true;
|
|
d->m_sticky = k->d->m_sticky;
|
|
d->m_mutable = k->d->m_mutable;
|
|
}
|
|
|
|
bool Kit::isValid() const
|
|
{
|
|
if (!d->m_id.isValid())
|
|
return false;
|
|
|
|
if (!d->m_hasValidityInfo)
|
|
validate();
|
|
|
|
return d->m_isValid;
|
|
}
|
|
|
|
bool Kit::hasWarning() const
|
|
{
|
|
if (!d->m_hasValidityInfo)
|
|
validate();
|
|
|
|
return d->m_hasWarning;
|
|
}
|
|
|
|
QList<Task> Kit::validate() const
|
|
{
|
|
QList<Task> result;
|
|
QList<KitInformation *> infoList = KitManager::kitInformation();
|
|
d->m_isValid = true;
|
|
d->m_hasWarning = false;
|
|
foreach (KitInformation *i, infoList) {
|
|
QList<Task> tmp = i->validate(this);
|
|
foreach (const Task &t, tmp) {
|
|
if (t.type == Task::Error)
|
|
d->m_isValid = false;
|
|
if (t.type == Task::Warning)
|
|
d->m_hasWarning = true;
|
|
}
|
|
result.append(tmp);
|
|
}
|
|
Utils::sort(result);
|
|
d->m_hasValidityInfo = true;
|
|
return result;
|
|
}
|
|
|
|
void Kit::fix()
|
|
{
|
|
KitGuard g(this);
|
|
foreach (KitInformation *i, KitManager::kitInformation())
|
|
i->fix(this);
|
|
}
|
|
|
|
void Kit::setup()
|
|
{
|
|
KitGuard g(this);
|
|
// Process the KitInfos in reverse order: They may only be based on other information lower in
|
|
// the stack.
|
|
QList<KitInformation *> info = KitManager::kitInformation();
|
|
for (int i = info.count() - 1; i >= 0; --i)
|
|
info.at(i)->setup(this);
|
|
}
|
|
|
|
QString Kit::unexpandedDisplayName() const
|
|
{
|
|
return d->m_unexpandedDisplayName;
|
|
}
|
|
|
|
QString Kit::displayName() const
|
|
{
|
|
return d->m_macroExpander.expand(d->m_unexpandedDisplayName);
|
|
}
|
|
|
|
static QString candidateName(const QString &name, const QString &postfix)
|
|
{
|
|
if (name.contains(postfix))
|
|
return QString();
|
|
QString candidate = name;
|
|
if (!candidate.isEmpty())
|
|
candidate.append(QLatin1Char('-'));
|
|
candidate.append(postfix);
|
|
return candidate;
|
|
}
|
|
|
|
void Kit::setUnexpandedDisplayName(const QString &name)
|
|
{
|
|
if (d->m_unexpandedDisplayName == name)
|
|
return;
|
|
|
|
d->m_unexpandedDisplayName = name;
|
|
kitUpdated();
|
|
}
|
|
|
|
QStringList Kit::candidateNameList(const QString &base) const
|
|
{
|
|
QStringList result;
|
|
result << base;
|
|
foreach (KitInformation *ki, KitManager::kitInformation()) {
|
|
const QString postfix = ki->displayNamePostfix(this);
|
|
if (!postfix.isEmpty()) {
|
|
QString tmp = candidateName(base, postfix);
|
|
if (!tmp.isEmpty())
|
|
result << tmp;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void Kit::setCustomFileSystemFriendlyName(const QString &fileSystemFriendlyName)
|
|
{
|
|
d->m_fileSystemFriendlyName = fileSystemFriendlyName;
|
|
}
|
|
|
|
QString Kit::customFileSystemFriendlyName() const
|
|
{
|
|
return d->m_fileSystemFriendlyName;
|
|
}
|
|
|
|
QString Kit::fileSystemFriendlyName() const
|
|
{
|
|
QString name = customFileSystemFriendlyName();
|
|
if (name.isEmpty())
|
|
name = FileUtils::qmakeFriendlyName(displayName());
|
|
foreach (Kit *i, KitManager::kits()) {
|
|
if (i == this)
|
|
continue;
|
|
if (name == FileUtils::qmakeFriendlyName(i->displayName())) {
|
|
// append part of the kit id: That should be unique enough;-)
|
|
// Leading { will be turned into _ which should be fine.
|
|
name = FileUtils::qmakeFriendlyName(name + QLatin1Char('_') + (id().toString().left(7)));
|
|
break;
|
|
}
|
|
}
|
|
return name;
|
|
}
|
|
|
|
bool Kit::isAutoDetected() const
|
|
{
|
|
return d->m_autodetected;
|
|
}
|
|
|
|
QString Kit::autoDetectionSource() const
|
|
{
|
|
return d->m_autoDetectionSource;
|
|
}
|
|
|
|
bool Kit::isSdkProvided() const
|
|
{
|
|
return d->m_sdkProvided;
|
|
}
|
|
|
|
Id Kit::id() const
|
|
{
|
|
return d->m_id;
|
|
}
|
|
|
|
QIcon Kit::icon() const
|
|
{
|
|
return d->m_icon;
|
|
}
|
|
|
|
QIcon Kit::icon(const FileName &path)
|
|
{
|
|
if (path.isEmpty())
|
|
return QIcon();
|
|
if (path == FileName::fromLatin1(":///DESKTOP///"))
|
|
return qApp->style()->standardIcon(QStyle::SP_ComputerIcon);
|
|
|
|
QFileInfo fi(path.toString());
|
|
if (fi.isFile() && fi.isReadable())
|
|
return QIcon(path.toString());
|
|
return QIcon();
|
|
}
|
|
|
|
FileName Kit::iconPath() const
|
|
{
|
|
return d->m_iconPath;
|
|
}
|
|
|
|
void Kit::setIconPath(const FileName &path)
|
|
{
|
|
if (d->m_iconPath == path)
|
|
return;
|
|
d->m_iconPath = path;
|
|
d->m_icon = icon(path);
|
|
kitUpdated();
|
|
}
|
|
|
|
QVariant Kit::value(Id key, const QVariant &unset) const
|
|
{
|
|
return d->m_data.value(key, unset);
|
|
}
|
|
|
|
bool Kit::hasValue(Id key) const
|
|
{
|
|
return d->m_data.contains(key);
|
|
}
|
|
|
|
void Kit::setValue(Id key, const QVariant &value)
|
|
{
|
|
if (d->m_data.value(key) == value)
|
|
return;
|
|
d->m_data.insert(key, value);
|
|
kitUpdated();
|
|
}
|
|
|
|
/// \internal
|
|
void Kit::setValueSilently(Id key, const QVariant &value)
|
|
{
|
|
if (d->m_data.value(key) == value)
|
|
return;
|
|
d->m_data.insert(key, value);
|
|
}
|
|
|
|
/// \internal
|
|
void Kit::removeKeySilently(Id key)
|
|
{
|
|
if (!d->m_data.contains(key))
|
|
return;
|
|
d->m_data.remove(key);
|
|
d->m_sticky.remove(key);
|
|
d->m_mutable.remove(key);
|
|
}
|
|
|
|
|
|
void Kit::removeKey(Id key)
|
|
{
|
|
if (!d->m_data.contains(key))
|
|
return;
|
|
d->m_data.remove(key);
|
|
d->m_sticky.remove(key);
|
|
d->m_mutable.remove(key);
|
|
kitUpdated();
|
|
}
|
|
|
|
bool Kit::isSticky(Core::Id id) const
|
|
{
|
|
return d->m_sticky.contains(id);
|
|
}
|
|
|
|
bool Kit::isDataEqual(const Kit *other) const
|
|
{
|
|
return d->m_data == other->d->m_data;
|
|
}
|
|
|
|
bool Kit::isEqual(const Kit *other) const
|
|
{
|
|
return isDataEqual(other)
|
|
&& d->m_iconPath == other->d->m_iconPath
|
|
&& d->m_unexpandedDisplayName == other->d->m_unexpandedDisplayName
|
|
&& d->m_fileSystemFriendlyName == other->d->m_fileSystemFriendlyName
|
|
&& d->m_mutable == other->d->m_mutable;
|
|
|
|
}
|
|
|
|
QVariantMap Kit::toMap() const
|
|
{
|
|
typedef QHash<Core::Id, QVariant>::ConstIterator IdVariantConstIt;
|
|
|
|
QVariantMap data;
|
|
data.insert(QLatin1String(ID_KEY), QString::fromLatin1(d->m_id.name()));
|
|
data.insert(QLatin1String(DISPLAYNAME_KEY), d->m_unexpandedDisplayName);
|
|
data.insert(QLatin1String(AUTODETECTED_KEY), d->m_autodetected);
|
|
if (!d->m_fileSystemFriendlyName.isEmpty())
|
|
data.insert(QLatin1String(FILESYSTEMFRIENDLYNAME_KEY), d->m_fileSystemFriendlyName);
|
|
data.insert(QLatin1String(AUTODETECTIONSOURCE_KEY), d->m_autoDetectionSource);
|
|
data.insert(QLatin1String(SDK_PROVIDED_KEY), d->m_sdkProvided);
|
|
data.insert(QLatin1String(ICON_KEY), d->m_iconPath.toString());
|
|
|
|
QStringList mutableInfo;
|
|
foreach (Core::Id id, d->m_mutable)
|
|
mutableInfo << id.toString();
|
|
data.insert(QLatin1String(MUTABLE_INFO_KEY), mutableInfo);
|
|
|
|
QStringList stickyInfo;
|
|
foreach (Core::Id id, d->m_sticky)
|
|
stickyInfo << id.toString();
|
|
data.insert(QLatin1String(STICKY_INFO_KEY), stickyInfo);
|
|
|
|
QVariantMap extra;
|
|
|
|
const IdVariantConstIt cend = d->m_data.constEnd();
|
|
for (IdVariantConstIt it = d->m_data.constBegin(); it != cend; ++it)
|
|
extra.insert(QString::fromLatin1(it.key().name().constData()), it.value());
|
|
data.insert(QLatin1String(DATA_KEY), extra);
|
|
|
|
return data;
|
|
}
|
|
|
|
void Kit::addToEnvironment(Environment &env) const
|
|
{
|
|
QList<KitInformation *> infoList = KitManager::kitInformation();
|
|
foreach (KitInformation *ki, infoList)
|
|
ki->addToEnvironment(this, env);
|
|
}
|
|
|
|
IOutputParser *Kit::createOutputParser() const
|
|
{
|
|
IOutputParser *first = new OsParser;
|
|
QList<KitInformation *> infoList = KitManager::kitInformation();
|
|
foreach (KitInformation *ki, infoList)
|
|
first->appendOutputParser(ki->createOutputParser(this));
|
|
return first;
|
|
}
|
|
|
|
QString Kit::toHtml(const QList<Task> &additional) const
|
|
{
|
|
QString rc;
|
|
QTextStream str(&rc);
|
|
str << "<html><body>";
|
|
str << "<h3>" << displayName() << "</h3>";
|
|
str << "<table>";
|
|
|
|
if (!isValid() || hasWarning() || !additional.isEmpty()) {
|
|
QList<Task> issues = additional;
|
|
issues.append(validate());
|
|
str << "<p>";
|
|
foreach (const Task &t, issues) {
|
|
str << "<b>";
|
|
switch (t.type) {
|
|
case Task::Error:
|
|
str << QCoreApplication::translate("ProjectExplorer::Kit", "Error:") << " ";
|
|
break;
|
|
case Task::Warning:
|
|
str << QCoreApplication::translate("ProjectExplorer::Kit", "Warning:") << " ";
|
|
break;
|
|
case Task::Unknown:
|
|
default:
|
|
break;
|
|
}
|
|
str << "</b>" << t.description << "<br>";
|
|
}
|
|
str << "</p>";
|
|
}
|
|
|
|
QList<KitInformation *> infoList = KitManager::kitInformation();
|
|
foreach (KitInformation *ki, infoList) {
|
|
KitInformation::ItemList list = ki->toUserOutput(this);
|
|
foreach (const KitInformation::Item &j, list)
|
|
str << "<tr><td><b>" << j.first << ":</b></td><td>" << j.second << "</td></tr>";
|
|
}
|
|
str << "</table></body></html>";
|
|
return rc;
|
|
}
|
|
|
|
void Kit::setAutoDetected(bool detected)
|
|
{
|
|
d->m_autodetected = detected;
|
|
}
|
|
|
|
void Kit::setAutoDetectionSource(const QString &autoDetectionSource)
|
|
{
|
|
d->m_autoDetectionSource = autoDetectionSource;
|
|
}
|
|
|
|
void Kit::setSdkProvided(bool sdkProvided)
|
|
{
|
|
d->m_sdkProvided = sdkProvided;
|
|
}
|
|
|
|
void Kit::makeSticky()
|
|
{
|
|
foreach (KitInformation *ki, KitManager::kitInformation()) {
|
|
if (hasValue(ki->id()))
|
|
setSticky(ki->id(), true);
|
|
}
|
|
}
|
|
|
|
void Kit::setSticky(Core::Id id, bool b)
|
|
{
|
|
if (b)
|
|
d->m_sticky.insert(id);
|
|
else
|
|
d->m_sticky.remove(id);
|
|
}
|
|
|
|
void Kit::makeUnSticky()
|
|
{
|
|
d->m_sticky.clear();
|
|
}
|
|
|
|
void Kit::setMutable(Id id, bool b)
|
|
{
|
|
if (b)
|
|
d->m_mutable.insert(id);
|
|
else
|
|
d->m_mutable.remove(id);
|
|
}
|
|
|
|
bool Kit::isMutable(Id id) const
|
|
{
|
|
return d->m_mutable.contains(id);
|
|
}
|
|
|
|
QSet<QString> Kit::availablePlatforms() const
|
|
{
|
|
QSet<QString> platforms;
|
|
foreach (const KitInformation *ki, KitManager::kitInformation())
|
|
platforms.unite(ki->availablePlatforms(this));
|
|
return platforms;
|
|
}
|
|
|
|
bool Kit::hasPlatform(const QString &platform) const
|
|
{
|
|
if (platform.isEmpty())
|
|
return true;
|
|
return availablePlatforms().contains(platform);
|
|
}
|
|
|
|
QString Kit::displayNameForPlatform(const QString &platform) const
|
|
{
|
|
foreach (const KitInformation *ki, KitManager::kitInformation()) {
|
|
const QString displayName = ki->displayNameForPlatform(this, platform);
|
|
if (!displayName.isEmpty())
|
|
return displayName;
|
|
}
|
|
return QString();
|
|
|
|
}
|
|
|
|
FeatureSet Kit::availableFeatures() const
|
|
{
|
|
Core::FeatureSet features;
|
|
foreach (const KitInformation *ki, KitManager::kitInformation())
|
|
features |= ki->availableFeatures(this);
|
|
return features;
|
|
}
|
|
|
|
bool Kit::hasFeatures(const FeatureSet &features) const
|
|
{
|
|
return availableFeatures().contains(features);
|
|
}
|
|
|
|
MacroExpander *Kit::macroExpander() const
|
|
{
|
|
return &d->m_macroExpander;
|
|
}
|
|
|
|
void Kit::kitUpdated()
|
|
{
|
|
if (d->m_nestedBlockingLevel > 0) {
|
|
d->m_mustNotify = true;
|
|
return;
|
|
}
|
|
d->m_hasValidityInfo = false;
|
|
KitManager::notifyAboutUpdate(this);
|
|
d->m_mustNotify = false;
|
|
}
|
|
|
|
} // namespace ProjectExplorer
|