forked from qt-creator/qt-creator
As for the build config, we provide a global variant for use by external
tools, and one for use in the respective run configuration itself.
Also hide the deprecated "CurrentRun" variant, which was forgotten in
b55825a420.
Fixes: QTCREATORBUG-25561
Change-Id: I51aaff10301f7ff1256abf1c09ac9f5be136ab00
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
938 lines
32 KiB
C++
938 lines
32 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** 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 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.
|
|
**
|
|
** 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.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "projectconfigurationmodel.h"
|
|
#include "target.h"
|
|
|
|
#include "buildconfiguration.h"
|
|
#include "buildinfo.h"
|
|
#include "buildmanager.h"
|
|
#include "buildsystem.h"
|
|
#include "buildtargetinfo.h"
|
|
#include "deployconfiguration.h"
|
|
#include "deploymentdata.h"
|
|
#include "devicesupport/devicemanager.h"
|
|
#include "environmentaspect.h"
|
|
#include "kit.h"
|
|
#include "kitinformation.h"
|
|
#include "kitmanager.h"
|
|
#include "miniprojecttargetselector.h"
|
|
#include "project.h"
|
|
#include "projectexplorer.h"
|
|
#include "projectexplorericons.h"
|
|
#include "projectexplorersettings.h"
|
|
#include "runconfiguration.h"
|
|
#include "runconfigurationaspects.h"
|
|
#include "session.h"
|
|
|
|
#include <coreplugin/coreconstants.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/macroexpander.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/stringutils.h>
|
|
|
|
#include <QDebug>
|
|
#include <QIcon>
|
|
#include <QPainter>
|
|
|
|
#include <limits>
|
|
|
|
using namespace Utils;
|
|
|
|
namespace ProjectExplorer {
|
|
|
|
const char ACTIVE_BC_KEY[] = "ProjectExplorer.Target.ActiveBuildConfiguration";
|
|
const char BC_KEY_PREFIX[] = "ProjectExplorer.Target.BuildConfiguration.";
|
|
const char BC_COUNT_KEY[] = "ProjectExplorer.Target.BuildConfigurationCount";
|
|
|
|
const char ACTIVE_DC_KEY[] = "ProjectExplorer.Target.ActiveDeployConfiguration";
|
|
const char DC_KEY_PREFIX[] = "ProjectExplorer.Target.DeployConfiguration.";
|
|
const char DC_COUNT_KEY[] = "ProjectExplorer.Target.DeployConfigurationCount";
|
|
|
|
const char ACTIVE_RC_KEY[] = "ProjectExplorer.Target.ActiveRunConfiguration";
|
|
const char RC_KEY_PREFIX[] = "ProjectExplorer.Target.RunConfiguration.";
|
|
const char RC_COUNT_KEY[] = "ProjectExplorer.Target.RunConfigurationCount";
|
|
const char PLUGIN_SETTINGS_KEY[] = "ProjectExplorer.Target.PluginSettings";
|
|
|
|
static QString formatDeviceInfo(const ProjectExplorer::IDevice::DeviceInfo &input)
|
|
{
|
|
const QStringList lines
|
|
= Utils::transform(input, [](const ProjectExplorer::IDevice::DeviceInfoItem &i) {
|
|
return QString::fromLatin1("<b>%1:</b> %2").arg(i.key, i.value);
|
|
});
|
|
return lines.join(QLatin1String("<br>"));
|
|
}
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Target
|
|
// -------------------------------------------------------------------------
|
|
|
|
class TargetPrivate
|
|
{
|
|
public:
|
|
TargetPrivate(Target *t, Kit *k) :
|
|
m_kit(k),
|
|
m_buildConfigurationModel(t),
|
|
m_deployConfigurationModel(t),
|
|
m_runConfigurationModel(t)
|
|
{ }
|
|
|
|
~TargetPrivate()
|
|
{
|
|
delete m_buildSystem;
|
|
}
|
|
|
|
QIcon m_overlayIcon;
|
|
|
|
QList<BuildConfiguration *> m_buildConfigurations;
|
|
QPointer<BuildConfiguration> m_activeBuildConfiguration;
|
|
QList<DeployConfiguration *> m_deployConfigurations;
|
|
DeployConfiguration *m_activeDeployConfiguration = nullptr;
|
|
QList<RunConfiguration *> m_runConfigurations;
|
|
RunConfiguration* m_activeRunConfiguration = nullptr;
|
|
QVariantMap m_pluginSettings;
|
|
|
|
Kit *const m_kit;
|
|
MacroExpander m_macroExpander;
|
|
BuildSystem *m_buildSystem = nullptr;
|
|
|
|
ProjectConfigurationModel m_buildConfigurationModel;
|
|
ProjectConfigurationModel m_deployConfigurationModel;
|
|
ProjectConfigurationModel m_runConfigurationModel;
|
|
};
|
|
|
|
|
|
Target::Target(Project *project, Kit *k, _constructor_tag) :
|
|
QObject(project),
|
|
d(std::make_unique<TargetPrivate>(this, k))
|
|
{
|
|
// Note: nullptr is a valid state for the per-buildConfig systems.
|
|
d->m_buildSystem = project->createBuildSystem(this);
|
|
|
|
QTC_CHECK(d->m_kit);
|
|
connect(DeviceManager::instance(), &DeviceManager::updated, this, &Target::updateDeviceState);
|
|
|
|
connect(this, &Target::parsingStarted, this, [this, project] {
|
|
project->anyParsingStarted(this);
|
|
});
|
|
|
|
connect(this, &Target::parsingFinished, this, [this, project](bool success) {
|
|
if (success && this == SessionManager::startupTarget())
|
|
updateDefaultRunConfigurations();
|
|
// For testing.
|
|
emit SessionManager::instance()->projectFinishedParsing(project);
|
|
project->anyParsingFinished(this, success);
|
|
}, Qt::QueuedConnection); // Must wait for run configs to change their enabled state.
|
|
|
|
KitManager *km = KitManager::instance();
|
|
connect(km, &KitManager::kitUpdated, this, &Target::handleKitUpdates);
|
|
connect(km, &KitManager::kitRemoved, this, &Target::handleKitRemoval);
|
|
|
|
d->m_macroExpander.setDisplayName(tr("Target Settings"));
|
|
d->m_macroExpander.setAccumulating(true);
|
|
|
|
d->m_macroExpander.registerSubProvider([this] { return kit()->macroExpander(); });
|
|
|
|
d->m_macroExpander.registerVariable("sourceDir", tr("Source directory"),
|
|
[project] { return project->projectDirectory().toUserOutput(); });
|
|
|
|
// TODO: Remove in ~4.16.
|
|
d->m_macroExpander.registerVariable(Constants::VAR_CURRENTPROJECT_NAME,
|
|
QCoreApplication::translate("ProjectExplorer", "Name of current project"),
|
|
[project] { return project->displayName(); },
|
|
false);
|
|
d->m_macroExpander.registerVariable("Project:Name",
|
|
QCoreApplication::translate("ProjectExplorer", "Name of current project"),
|
|
[project] { return project->displayName(); });
|
|
|
|
d->m_macroExpander.registerVariable("CurrentRun:Name",
|
|
tr("The currently active run configuration's name."),
|
|
[this]() -> QString {
|
|
if (RunConfiguration * const rc = activeRunConfiguration())
|
|
return rc->displayName();
|
|
return QString();
|
|
}, false);
|
|
d->m_macroExpander.registerFileVariables("CurrentRun:Executable",
|
|
tr("The currently active run configuration's executable (if applicable)."),
|
|
[this]() -> QString {
|
|
if (RunConfiguration * const rc = activeRunConfiguration())
|
|
return rc->commandLine().executable().toString();
|
|
return QString();
|
|
}, false);
|
|
d->m_macroExpander.registerPrefix("CurrentRun:Env", tr("Variables in the current run environment."),
|
|
[this](const QString &var) {
|
|
if (RunConfiguration * const rc = activeRunConfiguration()) {
|
|
if (const auto envAspect = rc->aspect<EnvironmentAspect>())
|
|
return envAspect->environment().expandedValueForKey(var);
|
|
}
|
|
return QString();
|
|
}, false);
|
|
d->m_macroExpander.registerVariable("CurrentRun:WorkingDir",
|
|
tr("The currently active run configuration's working directory."),
|
|
[this] {
|
|
if (RunConfiguration * const rc = activeRunConfiguration()) {
|
|
if (const auto wdAspect = rc->aspect<WorkingDirectoryAspect>())
|
|
return wdAspect->workingDirectory(&d->m_macroExpander).toString();
|
|
}
|
|
return QString();
|
|
}, false);
|
|
}
|
|
|
|
Target::~Target()
|
|
{
|
|
qDeleteAll(d->m_buildConfigurations);
|
|
qDeleteAll(d->m_deployConfigurations);
|
|
qDeleteAll(d->m_runConfigurations);
|
|
}
|
|
|
|
void Target::handleKitUpdates(Kit *k)
|
|
{
|
|
if (k != d->m_kit)
|
|
return;
|
|
|
|
updateDefaultDeployConfigurations();
|
|
updateDeviceState(); // in case the device changed...
|
|
|
|
emit iconChanged();
|
|
emit kitChanged();
|
|
}
|
|
|
|
void Target::handleKitRemoval(Kit *k)
|
|
{
|
|
if (k != d->m_kit)
|
|
return;
|
|
project()->removeTarget(this);
|
|
}
|
|
|
|
bool Target::isActive() const
|
|
{
|
|
return project()->activeTarget() == this;
|
|
}
|
|
|
|
Project *Target::project() const
|
|
{
|
|
return static_cast<Project *>(parent());
|
|
}
|
|
|
|
Kit *Target::kit() const
|
|
{
|
|
return d->m_kit;
|
|
}
|
|
|
|
BuildSystem *Target::buildSystem() const
|
|
{
|
|
if (d->m_activeBuildConfiguration)
|
|
return d->m_activeBuildConfiguration->buildSystem();
|
|
|
|
return d->m_buildSystem;
|
|
}
|
|
|
|
BuildSystem *Target::fallbackBuildSystem() const
|
|
{
|
|
return d->m_buildSystem;
|
|
}
|
|
|
|
DeploymentData Target::deploymentData() const
|
|
{
|
|
const DeployConfiguration * const dc = activeDeployConfiguration();
|
|
if (dc && dc->usesCustomDeploymentData())
|
|
return dc->customDeploymentData();
|
|
return buildSystemDeploymentData();
|
|
}
|
|
|
|
DeploymentData Target::buildSystemDeploymentData() const
|
|
{
|
|
QTC_ASSERT(buildSystem(), return {});
|
|
return buildSystem()->deploymentData();
|
|
}
|
|
|
|
BuildTargetInfo Target::buildTarget(const QString &buildKey) const
|
|
{
|
|
QTC_ASSERT(buildSystem(), return {});
|
|
return buildSystem()->buildTarget(buildKey);
|
|
}
|
|
|
|
QString Target::activeBuildKey() const
|
|
{
|
|
// Should not happen. If it does, return a buildKey that wont be found in
|
|
// the project tree, so that the project()->findNodeForBuildKey(buildKey)
|
|
// returns null.
|
|
QTC_ASSERT(d->m_activeRunConfiguration, return QString(QChar(0)));
|
|
return d->m_activeRunConfiguration->buildKey();
|
|
}
|
|
|
|
Utils::Id Target::id() const
|
|
{
|
|
return d->m_kit->id();
|
|
}
|
|
|
|
QString Target::displayName() const
|
|
{
|
|
return d->m_kit->displayName();
|
|
}
|
|
|
|
QString Target::toolTip() const
|
|
{
|
|
return d->m_kit->toHtml();
|
|
}
|
|
|
|
QString Target::displayNameKey()
|
|
{
|
|
return QString("ProjectExplorer.ProjectConfiguration.DisplayName");
|
|
}
|
|
|
|
QString Target::deviceTypeKey()
|
|
{
|
|
return QString("DeviceType");
|
|
}
|
|
|
|
void Target::addBuildConfiguration(BuildConfiguration *bc)
|
|
{
|
|
QTC_ASSERT(bc && !d->m_buildConfigurations.contains(bc), return);
|
|
Q_ASSERT(bc->target() == this);
|
|
|
|
// Check that we don't have a configuration with the same displayName
|
|
QString configurationDisplayName = bc->displayName();
|
|
QStringList displayNames = Utils::transform(d->m_buildConfigurations, &BuildConfiguration::displayName);
|
|
configurationDisplayName = Utils::makeUniquelyNumbered(configurationDisplayName, displayNames);
|
|
if (configurationDisplayName != bc->displayName()) {
|
|
if (bc->usesDefaultDisplayName())
|
|
bc->setDefaultDisplayName(configurationDisplayName);
|
|
else
|
|
bc->setDisplayName(configurationDisplayName);
|
|
}
|
|
|
|
// add it
|
|
d->m_buildConfigurations.push_back(bc);
|
|
|
|
ProjectExplorerPlugin::targetSelector()->addedBuildConfiguration(bc);
|
|
emit addedBuildConfiguration(bc);
|
|
d->m_buildConfigurationModel.addProjectConfiguration(bc);
|
|
|
|
if (!activeBuildConfiguration())
|
|
setActiveBuildConfiguration(bc);
|
|
}
|
|
|
|
bool Target::removeBuildConfiguration(BuildConfiguration *bc)
|
|
{
|
|
//todo: this might be error prone
|
|
if (!d->m_buildConfigurations.contains(bc))
|
|
return false;
|
|
|
|
if (BuildManager::isBuilding(bc))
|
|
return false;
|
|
|
|
d->m_buildConfigurations.removeOne(bc);
|
|
|
|
if (activeBuildConfiguration() == bc) {
|
|
if (d->m_buildConfigurations.isEmpty())
|
|
SessionManager::setActiveBuildConfiguration(this, nullptr, SetActive::Cascade);
|
|
else
|
|
SessionManager::setActiveBuildConfiguration(this, d->m_buildConfigurations.at(0), SetActive::Cascade);
|
|
}
|
|
|
|
emit removedBuildConfiguration(bc);
|
|
ProjectExplorerPlugin::targetSelector()->removedBuildConfiguration(bc);
|
|
d->m_buildConfigurationModel.removeProjectConfiguration(bc);
|
|
|
|
delete bc;
|
|
return true;
|
|
}
|
|
|
|
const QList<BuildConfiguration *> Target::buildConfigurations() const
|
|
{
|
|
return d->m_buildConfigurations;
|
|
}
|
|
|
|
BuildConfiguration *Target::activeBuildConfiguration() const
|
|
{
|
|
return d->m_activeBuildConfiguration;
|
|
}
|
|
|
|
void Target::setActiveBuildConfiguration(BuildConfiguration *bc)
|
|
{
|
|
if ((!bc && d->m_buildConfigurations.isEmpty()) ||
|
|
(bc && d->m_buildConfigurations.contains(bc) &&
|
|
bc != d->m_activeBuildConfiguration)) {
|
|
d->m_activeBuildConfiguration = bc;
|
|
emit activeBuildConfigurationChanged(d->m_activeBuildConfiguration);
|
|
ProjectExplorerPlugin::updateActions();
|
|
}
|
|
}
|
|
|
|
void Target::addDeployConfiguration(DeployConfiguration *dc)
|
|
{
|
|
QTC_ASSERT(dc && !d->m_deployConfigurations.contains(dc), return);
|
|
Q_ASSERT(dc->target() == this);
|
|
|
|
// Check that we don't have a configuration with the same displayName
|
|
QString configurationDisplayName = dc->displayName();
|
|
QStringList displayNames = Utils::transform(d->m_deployConfigurations, &DeployConfiguration::displayName);
|
|
configurationDisplayName = Utils::makeUniquelyNumbered(configurationDisplayName, displayNames);
|
|
dc->setDisplayName(configurationDisplayName);
|
|
|
|
// add it
|
|
d->m_deployConfigurations.push_back(dc);
|
|
|
|
ProjectExplorerPlugin::targetSelector()->addedDeployConfiguration(dc);
|
|
d->m_deployConfigurationModel.addProjectConfiguration(dc);
|
|
emit addedDeployConfiguration(dc);
|
|
|
|
if (!d->m_activeDeployConfiguration)
|
|
setActiveDeployConfiguration(dc);
|
|
Q_ASSERT(activeDeployConfiguration());
|
|
}
|
|
|
|
bool Target::removeDeployConfiguration(DeployConfiguration *dc)
|
|
{
|
|
//todo: this might be error prone
|
|
if (!d->m_deployConfigurations.contains(dc))
|
|
return false;
|
|
|
|
if (BuildManager::isBuilding(dc))
|
|
return false;
|
|
|
|
d->m_deployConfigurations.removeOne(dc);
|
|
|
|
if (activeDeployConfiguration() == dc) {
|
|
if (d->m_deployConfigurations.isEmpty())
|
|
SessionManager::setActiveDeployConfiguration(this, nullptr, SetActive::Cascade);
|
|
else
|
|
SessionManager::setActiveDeployConfiguration(this, d->m_deployConfigurations.at(0),
|
|
SetActive::Cascade);
|
|
}
|
|
|
|
ProjectExplorerPlugin::targetSelector()->removedDeployConfiguration(dc);
|
|
d->m_deployConfigurationModel.removeProjectConfiguration(dc);
|
|
emit removedDeployConfiguration(dc);
|
|
|
|
delete dc;
|
|
return true;
|
|
}
|
|
|
|
const QList<DeployConfiguration *> Target::deployConfigurations() const
|
|
{
|
|
return d->m_deployConfigurations;
|
|
}
|
|
|
|
DeployConfiguration *Target::activeDeployConfiguration() const
|
|
{
|
|
return d->m_activeDeployConfiguration;
|
|
}
|
|
|
|
void Target::setActiveDeployConfiguration(DeployConfiguration *dc)
|
|
{
|
|
if ((!dc && d->m_deployConfigurations.isEmpty()) ||
|
|
(dc && d->m_deployConfigurations.contains(dc) &&
|
|
dc != d->m_activeDeployConfiguration)) {
|
|
d->m_activeDeployConfiguration = dc;
|
|
emit activeDeployConfigurationChanged(d->m_activeDeployConfiguration);
|
|
}
|
|
updateDeviceState();
|
|
}
|
|
|
|
const QList<RunConfiguration *> Target::runConfigurations() const
|
|
{
|
|
return d->m_runConfigurations;
|
|
}
|
|
|
|
void Target::addRunConfiguration(RunConfiguration *rc)
|
|
{
|
|
QTC_ASSERT(rc && !d->m_runConfigurations.contains(rc), return);
|
|
Q_ASSERT(rc->target() == this);
|
|
|
|
// Check that we don't have a configuration with the same displayName
|
|
QString configurationDisplayName = rc->displayName();
|
|
if (!configurationDisplayName.isEmpty()) {
|
|
QStringList displayNames = Utils::transform(d->m_runConfigurations,
|
|
&RunConfiguration::displayName);
|
|
configurationDisplayName = Utils::makeUniquelyNumbered(configurationDisplayName,
|
|
displayNames);
|
|
rc->setDisplayName(configurationDisplayName);
|
|
}
|
|
|
|
d->m_runConfigurations.push_back(rc);
|
|
|
|
ProjectExplorerPlugin::targetSelector()->addedRunConfiguration(rc);
|
|
d->m_runConfigurationModel.addProjectConfiguration(rc);
|
|
emit addedRunConfiguration(rc);
|
|
|
|
if (!activeRunConfiguration())
|
|
setActiveRunConfiguration(rc);
|
|
}
|
|
|
|
void Target::removeRunConfiguration(RunConfiguration *rc)
|
|
{
|
|
QTC_ASSERT(rc && d->m_runConfigurations.contains(rc), return);
|
|
|
|
d->m_runConfigurations.removeOne(rc);
|
|
|
|
if (activeRunConfiguration() == rc) {
|
|
if (d->m_runConfigurations.isEmpty())
|
|
setActiveRunConfiguration(nullptr);
|
|
else
|
|
setActiveRunConfiguration(d->m_runConfigurations.at(0));
|
|
}
|
|
|
|
emit removedRunConfiguration(rc);
|
|
ProjectExplorerPlugin::targetSelector()->removedRunConfiguration(rc);
|
|
d->m_runConfigurationModel.removeProjectConfiguration(rc);
|
|
|
|
delete rc;
|
|
}
|
|
|
|
RunConfiguration *Target::activeRunConfiguration() const
|
|
{
|
|
return d->m_activeRunConfiguration;
|
|
}
|
|
|
|
void Target::setActiveRunConfiguration(RunConfiguration *rc)
|
|
{
|
|
if ((!rc && d->m_runConfigurations.isEmpty()) ||
|
|
(rc && d->m_runConfigurations.contains(rc) &&
|
|
rc != d->m_activeRunConfiguration)) {
|
|
d->m_activeRunConfiguration = rc;
|
|
emit activeRunConfigurationChanged(d->m_activeRunConfiguration);
|
|
ProjectExplorerPlugin::updateActions();
|
|
}
|
|
updateDeviceState();
|
|
}
|
|
|
|
QIcon Target::icon() const
|
|
{
|
|
return d->m_kit->icon();
|
|
}
|
|
|
|
QIcon Target::overlayIcon() const
|
|
{
|
|
return d->m_overlayIcon;
|
|
}
|
|
|
|
void Target::setOverlayIcon(const QIcon &icon)
|
|
{
|
|
d->m_overlayIcon = icon;
|
|
emit overlayIconChanged();
|
|
}
|
|
|
|
QString Target::overlayIconToolTip()
|
|
{
|
|
IDevice::ConstPtr current = DeviceKitAspect::device(kit());
|
|
return current.isNull() ? QString() : formatDeviceInfo(current->deviceInformation());
|
|
}
|
|
|
|
QVariantMap Target::toMap() const
|
|
{
|
|
if (!d->m_kit) // Kit was deleted, target is only around to be copied.
|
|
return QVariantMap();
|
|
|
|
QVariantMap map;
|
|
map.insert(displayNameKey(), displayName());
|
|
map.insert(deviceTypeKey(), DeviceTypeKitAspect::deviceTypeId(kit()).toSetting());
|
|
|
|
{
|
|
// FIXME: For compatibility within the 4.11 cycle, remove this block later.
|
|
// This is only read by older versions of Creator, but even there not actively used.
|
|
const char CONFIGURATION_ID_KEY[] = "ProjectExplorer.ProjectConfiguration.Id";
|
|
const char DEFAULT_DISPLAY_NAME_KEY[] = "ProjectExplorer.ProjectConfiguration.DefaultDisplayName";
|
|
map.insert(QLatin1String(CONFIGURATION_ID_KEY), id().toSetting());
|
|
map.insert(QLatin1String(DEFAULT_DISPLAY_NAME_KEY), displayName());
|
|
}
|
|
|
|
const QList<BuildConfiguration *> bcs = buildConfigurations();
|
|
map.insert(QLatin1String(ACTIVE_BC_KEY), bcs.indexOf(d->m_activeBuildConfiguration));
|
|
map.insert(QLatin1String(BC_COUNT_KEY), bcs.size());
|
|
for (int i = 0; i < bcs.size(); ++i)
|
|
map.insert(QString::fromLatin1(BC_KEY_PREFIX) + QString::number(i), bcs.at(i)->toMap());
|
|
|
|
const QList<DeployConfiguration *> dcs = deployConfigurations();
|
|
map.insert(QLatin1String(ACTIVE_DC_KEY), dcs.indexOf(d->m_activeDeployConfiguration));
|
|
map.insert(QLatin1String(DC_COUNT_KEY), dcs.size());
|
|
for (int i = 0; i < dcs.size(); ++i)
|
|
map.insert(QString::fromLatin1(DC_KEY_PREFIX) + QString::number(i), dcs.at(i)->toMap());
|
|
|
|
const QList<RunConfiguration *> rcs = runConfigurations();
|
|
map.insert(QLatin1String(ACTIVE_RC_KEY), rcs.indexOf(d->m_activeRunConfiguration));
|
|
map.insert(QLatin1String(RC_COUNT_KEY), rcs.size());
|
|
for (int i = 0; i < rcs.size(); ++i)
|
|
map.insert(QString::fromLatin1(RC_KEY_PREFIX) + QString::number(i), rcs.at(i)->toMap());
|
|
|
|
if (!d->m_pluginSettings.isEmpty())
|
|
map.insert(QLatin1String(PLUGIN_SETTINGS_KEY), d->m_pluginSettings);
|
|
|
|
return map;
|
|
}
|
|
|
|
void Target::updateDefaultBuildConfigurations()
|
|
{
|
|
BuildConfigurationFactory *bcFactory = BuildConfigurationFactory::find(this);
|
|
if (!bcFactory) {
|
|
qWarning("No build configuration factory found for target id '%s'.", qPrintable(id().toString()));
|
|
return;
|
|
}
|
|
for (const BuildInfo &info : bcFactory->allAvailableSetups(kit(), project()->projectFilePath())) {
|
|
if (BuildConfiguration *bc = bcFactory->create(this, info))
|
|
addBuildConfiguration(bc);
|
|
}
|
|
}
|
|
|
|
void Target::updateDefaultDeployConfigurations()
|
|
{
|
|
QList<DeployConfigurationFactory *> dcFactories = DeployConfigurationFactory::find(this);
|
|
if (dcFactories.isEmpty()) {
|
|
qWarning("No deployment configuration factory found for target id '%s'.", qPrintable(id().toString()));
|
|
return;
|
|
}
|
|
|
|
QList<Utils::Id> dcIds;
|
|
foreach (DeployConfigurationFactory *dcFactory, dcFactories)
|
|
dcIds.append(dcFactory->creationId());
|
|
|
|
QList<DeployConfiguration *> dcList = deployConfigurations();
|
|
QList<Utils::Id> toCreate = dcIds;
|
|
|
|
foreach (DeployConfiguration *dc, dcList) {
|
|
if (dcIds.contains(dc->id()))
|
|
toCreate.removeOne(dc->id());
|
|
else
|
|
removeDeployConfiguration(dc);
|
|
}
|
|
|
|
foreach (Utils::Id id, toCreate) {
|
|
foreach (DeployConfigurationFactory *dcFactory, dcFactories) {
|
|
if (dcFactory->creationId() == id) {
|
|
DeployConfiguration *dc = dcFactory->create(this);
|
|
if (dc) {
|
|
QTC_CHECK(dc->id() == id);
|
|
addDeployConfiguration(dc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Target::updateDefaultRunConfigurations()
|
|
{
|
|
// Manual and Auto
|
|
const QList<RunConfigurationCreationInfo> creators
|
|
= RunConfigurationFactory::creatorsForTarget(this);
|
|
|
|
if (creators.isEmpty()) {
|
|
qWarning("No run configuration factory found for target id '%s'.", qPrintable(id().toString()));
|
|
return;
|
|
}
|
|
|
|
QList<RunConfiguration *> existingConfigured; // Existing configured RCs
|
|
QList<RunConfiguration *> existingUnconfigured; // Existing unconfigured RCs
|
|
QList<RunConfiguration *> newConfigured; // NEW configured Rcs
|
|
QList<RunConfiguration *> newUnconfigured; // NEW unconfigured RCs
|
|
|
|
// sort existing RCs into configured/unconfigured.
|
|
std::tie(existingConfigured, existingUnconfigured)
|
|
= Utils::partition(runConfigurations(),
|
|
[](const RunConfiguration *rc) { return rc->isConfigured(); });
|
|
int configuredCount = existingConfigured.count();
|
|
|
|
// Put outdated RCs into toRemove, do not bother with factories
|
|
// that produce already existing RCs
|
|
QList<RunConfiguration *> toRemove;
|
|
QList<RunConfigurationCreationInfo> existing;
|
|
foreach (RunConfiguration *rc, existingConfigured) {
|
|
bool present = false;
|
|
for (const RunConfigurationCreationInfo &item : creators) {
|
|
QString buildKey = rc->buildKey();
|
|
if (item.factory->runConfigurationId() == rc->id() && item.buildKey == buildKey) {
|
|
existing.append(item);
|
|
present = true;
|
|
}
|
|
}
|
|
if (!present && project()->knowsAllBuildExecutables())
|
|
toRemove.append(rc);
|
|
}
|
|
configuredCount -= toRemove.count();
|
|
|
|
bool removeExistingUnconfigured = false;
|
|
if (ProjectExplorerPlugin::projectExplorerSettings().automaticallyCreateRunConfigurations) {
|
|
// Create new "automatic" RCs and put them into newConfigured/newUnconfigured
|
|
foreach (const RunConfigurationCreationInfo &item, creators) {
|
|
if (item.creationMode == RunConfigurationCreationInfo::ManualCreationOnly)
|
|
continue;
|
|
bool exists = false;
|
|
for (const RunConfigurationCreationInfo &ex : existing) {
|
|
if (ex.factory == item.factory && ex.buildKey == item.buildKey)
|
|
exists = true;
|
|
}
|
|
if (exists)
|
|
continue;
|
|
|
|
RunConfiguration *rc = item.create(this);
|
|
if (!rc)
|
|
continue;
|
|
QTC_CHECK(rc->id() == item.factory->runConfigurationId());
|
|
if (!rc->isConfigured())
|
|
newUnconfigured << rc;
|
|
else
|
|
newConfigured << rc;
|
|
}
|
|
configuredCount += newConfigured.count();
|
|
|
|
// Decide what to do with the different categories:
|
|
if (configuredCount > 0) {
|
|
// new non-Custom Executable RCs were added
|
|
removeExistingUnconfigured = true;
|
|
qDeleteAll(newUnconfigured);
|
|
newUnconfigured.clear();
|
|
} else {
|
|
// no new RCs, use old or new CERCs?
|
|
if (!existingUnconfigured.isEmpty()) {
|
|
qDeleteAll(newUnconfigured);
|
|
newUnconfigured.clear();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Do actual changes:
|
|
foreach (RunConfiguration *rc, newConfigured)
|
|
addRunConfiguration(rc);
|
|
foreach (RunConfiguration *rc, newUnconfigured)
|
|
addRunConfiguration(rc);
|
|
|
|
// Generate complete list of RCs to remove later:
|
|
QList<RunConfiguration *> removalList;
|
|
foreach (RunConfiguration *rc, toRemove) {
|
|
removalList << rc;
|
|
existingConfigured.removeOne(rc); // make sure to also remove them from existingConfigured!
|
|
}
|
|
|
|
if (removeExistingUnconfigured) {
|
|
removalList.append(existingUnconfigured);
|
|
existingUnconfigured.clear();
|
|
}
|
|
|
|
// Make sure a configured RC will be active after we delete the RCs:
|
|
RunConfiguration *active = activeRunConfiguration();
|
|
if (active && (removalList.contains(active) || !active->isEnabled())) {
|
|
RunConfiguration *newConfiguredDefault = newConfigured.isEmpty() ? nullptr : newConfigured.at(0);
|
|
|
|
RunConfiguration *rc
|
|
= Utils::findOrDefault(existingConfigured,
|
|
[](RunConfiguration *rc) { return rc->isEnabled(); });
|
|
if (!rc) {
|
|
rc = Utils::findOr(newConfigured, newConfiguredDefault,
|
|
Utils::equal(&RunConfiguration::displayName, project()->displayName()));
|
|
}
|
|
if (!rc)
|
|
rc = newUnconfigured.isEmpty() ? nullptr : newUnconfigured.at(0);
|
|
if (!rc) {
|
|
// No RCs will be deleted, so use the one that will emit the minimum number of signals.
|
|
// One signal will be emitted from the next setActiveRunConfiguration, another one
|
|
// when the RC gets removed (and the activeRunConfiguration turns into a nullptr).
|
|
rc = removalList.isEmpty() ? nullptr : removalList.last();
|
|
}
|
|
|
|
if (rc)
|
|
setActiveRunConfiguration(rc);
|
|
}
|
|
|
|
// Remove the RCs that are no longer needed:
|
|
foreach (RunConfiguration *rc, removalList)
|
|
removeRunConfiguration(rc);
|
|
}
|
|
|
|
QVariant Target::namedSettings(const QString &name) const
|
|
{
|
|
return d->m_pluginSettings.value(name);
|
|
}
|
|
|
|
void Target::setNamedSettings(const QString &name, const QVariant &value)
|
|
{
|
|
if (value.isNull())
|
|
d->m_pluginSettings.remove(name);
|
|
else
|
|
d->m_pluginSettings.insert(name, value);
|
|
}
|
|
|
|
QVariant Target::additionalData(Utils::Id id) const
|
|
{
|
|
if (const BuildSystem *bs = buildSystem())
|
|
return bs->additionalData(id);
|
|
|
|
return {};
|
|
}
|
|
|
|
MakeInstallCommand Target::makeInstallCommand(const QString &installRoot) const
|
|
{
|
|
return project()->makeInstallCommand(this, installRoot);
|
|
}
|
|
|
|
MacroExpander *Target::macroExpander() const
|
|
{
|
|
return &d->m_macroExpander;
|
|
}
|
|
|
|
ProjectConfigurationModel *Target::buildConfigurationModel() const
|
|
{
|
|
return &d->m_buildConfigurationModel;
|
|
}
|
|
|
|
ProjectConfigurationModel *Target::deployConfigurationModel() const
|
|
{
|
|
return &d->m_deployConfigurationModel;
|
|
}
|
|
|
|
ProjectConfigurationModel *Target::runConfigurationModel() const
|
|
{
|
|
return &d->m_runConfigurationModel;
|
|
}
|
|
|
|
void Target::updateDeviceState()
|
|
{
|
|
IDevice::ConstPtr current = DeviceKitAspect::device(kit());
|
|
|
|
QIcon overlay;
|
|
static const QIcon disconnected = Icons::DEVICE_DISCONNECTED_INDICATOR_OVERLAY.icon();
|
|
if (current.isNull()) {
|
|
overlay = disconnected;
|
|
} else {
|
|
switch (current->deviceState()) {
|
|
case IDevice::DeviceStateUnknown:
|
|
overlay = QIcon();
|
|
return;
|
|
case IDevice::DeviceReadyToUse: {
|
|
static const QIcon ready = Icons::DEVICE_READY_INDICATOR_OVERLAY.icon();
|
|
overlay = ready;
|
|
break;
|
|
}
|
|
case IDevice::DeviceConnected: {
|
|
static const QIcon connected = Icons::DEVICE_CONNECTED_INDICATOR_OVERLAY.icon();
|
|
overlay = connected;
|
|
break;
|
|
}
|
|
case IDevice::DeviceDisconnected:
|
|
overlay = disconnected;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
setOverlayIcon(overlay);
|
|
}
|
|
|
|
bool Target::fromMap(const QVariantMap &map)
|
|
{
|
|
QTC_ASSERT(d->m_kit == KitManager::kit(id()), return false);
|
|
|
|
bool ok;
|
|
int bcCount = map.value(QLatin1String(BC_COUNT_KEY), 0).toInt(&ok);
|
|
if (!ok || bcCount < 0)
|
|
bcCount = 0;
|
|
int activeConfiguration = map.value(QLatin1String(ACTIVE_BC_KEY), 0).toInt(&ok);
|
|
if (!ok || activeConfiguration < 0)
|
|
activeConfiguration = 0;
|
|
if (0 > activeConfiguration || bcCount < activeConfiguration)
|
|
activeConfiguration = 0;
|
|
|
|
for (int i = 0; i < bcCount; ++i) {
|
|
const QString key = QString::fromLatin1(BC_KEY_PREFIX) + QString::number(i);
|
|
if (!map.contains(key))
|
|
return false;
|
|
const QVariantMap valueMap = map.value(key).toMap();
|
|
BuildConfiguration *bc = BuildConfigurationFactory::restore(this, valueMap);
|
|
if (!bc) {
|
|
qWarning("No factory found to restore build configuration!");
|
|
continue;
|
|
}
|
|
QTC_CHECK(bc->id() == ProjectExplorer::idFromMap(valueMap));
|
|
addBuildConfiguration(bc);
|
|
if (i == activeConfiguration)
|
|
setActiveBuildConfiguration(bc);
|
|
}
|
|
if (buildConfigurations().isEmpty() && BuildConfigurationFactory::find(this))
|
|
return false;
|
|
|
|
int dcCount = map.value(QLatin1String(DC_COUNT_KEY), 0).toInt(&ok);
|
|
if (!ok || dcCount < 0)
|
|
dcCount = 0;
|
|
activeConfiguration = map.value(QLatin1String(ACTIVE_DC_KEY), 0).toInt(&ok);
|
|
if (!ok || activeConfiguration < 0)
|
|
activeConfiguration = 0;
|
|
if (0 > activeConfiguration || dcCount < activeConfiguration)
|
|
activeConfiguration = 0;
|
|
|
|
for (int i = 0; i < dcCount; ++i) {
|
|
const QString key = QString::fromLatin1(DC_KEY_PREFIX) + QString::number(i);
|
|
if (!map.contains(key))
|
|
return false;
|
|
QVariantMap valueMap = map.value(key).toMap();
|
|
DeployConfiguration *dc = DeployConfigurationFactory::restore(this, valueMap);
|
|
if (!dc) {
|
|
Utils::Id id = idFromMap(valueMap);
|
|
qWarning("No factory found to restore deployment configuration of id '%s'!",
|
|
id.isValid() ? qPrintable(id.toString()) : "UNKNOWN");
|
|
continue;
|
|
}
|
|
QTC_CHECK(dc->id() == ProjectExplorer::idFromMap(valueMap));
|
|
addDeployConfiguration(dc);
|
|
if (i == activeConfiguration)
|
|
setActiveDeployConfiguration(dc);
|
|
}
|
|
|
|
int rcCount = map.value(QLatin1String(RC_COUNT_KEY), 0).toInt(&ok);
|
|
if (!ok || rcCount < 0)
|
|
rcCount = 0;
|
|
activeConfiguration = map.value(QLatin1String(ACTIVE_RC_KEY), 0).toInt(&ok);
|
|
if (!ok || activeConfiguration < 0)
|
|
activeConfiguration = 0;
|
|
if (0 > activeConfiguration || rcCount < activeConfiguration)
|
|
activeConfiguration = 0;
|
|
|
|
for (int i = 0; i < rcCount; ++i) {
|
|
const QString key = QString::fromLatin1(RC_KEY_PREFIX) + QString::number(i);
|
|
if (!map.contains(key))
|
|
return false;
|
|
|
|
// Ignore missing RCs: We will just populate them using the default ones.
|
|
QVariantMap valueMap = map.value(key).toMap();
|
|
RunConfiguration *rc = RunConfigurationFactory::restore(this, valueMap);
|
|
if (!rc)
|
|
continue;
|
|
const Utils::Id theIdFromMap = ProjectExplorer::idFromMap(valueMap);
|
|
if (!theIdFromMap.toString().contains("///::///")) { // Hack for cmake 4.10 -> 4.11
|
|
QTC_CHECK(rc->id().withSuffix(rc->buildKey()) == theIdFromMap);
|
|
}
|
|
addRunConfiguration(rc);
|
|
if (i == activeConfiguration)
|
|
setActiveRunConfiguration(rc);
|
|
}
|
|
|
|
if (map.contains(QLatin1String(PLUGIN_SETTINGS_KEY)))
|
|
d->m_pluginSettings = map.value(QLatin1String(PLUGIN_SETTINGS_KEY)).toMap();
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace ProjectExplorer
|