Files
qt-creator/src/plugins/projectexplorer/targetsettingspanel.cpp

683 lines
23 KiB
C++
Raw Normal View History

/****************************************************************************
**
** 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://qt.digia.com/licensing. For further information
** use the contact form at http://qt.digia.com/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 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
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#include "targetsettingspanel.h"
#include "buildinfo.h"
#include "buildsettingspropertiespage.h"
#include "ipotentialkit.h"
#include "kitoptionspage.h"
#include "project.h"
#include "projectimporter.h"
#include "projectwindow.h"
#include "runsettingspropertiespage.h"
#include "target.h"
#include "targetsettingswidget.h"
#include <coreplugin/icore.h>
#include <extensionsystem/pluginmanager.h>
#include <projectexplorer/projectexplorer.h>
#include <projectexplorer/kit.h>
#include <projectexplorer/kitmanager.h>
#include <projectexplorer/buildmanager.h>
#include <projectexplorer/buildconfiguration.h>
#include <projectexplorer/deployconfiguration.h>
#include <projectexplorer/runconfiguration.h>
#include <utils/qtcassert.h>
#include <QFileDialog>
#include <QLabel>
#include <QMenu>
#include <QMessageBox>
#include <QPushButton>
#include <QStackedWidget>
#include <QToolTip>
#include <QVBoxLayout>
using namespace Core;
namespace ProjectExplorer {
namespace Internal {
///
// TargetSettingsWidget
///
TargetSettingsPanelWidget::TargetSettingsPanelWidget(Project *project) :
m_currentTarget(0),
m_project(project),
m_importer(project->createProjectImporter()),
m_selector(0),
m_centralWidget(0),
m_changeMenu(0),
m_duplicateMenu(0),
m_lastAction(0),
m_importAction(0)
{
Q_ASSERT(m_project);
m_panelWidgets[0] = 0;
m_panelWidgets[1] = 0;
m_addMenu = new QMenu(this);
m_targetMenu = new QMenu(this);
if (m_importer) {
m_importAction = new QAction(tr("Import existing build..."), this);
connect(m_importAction, SIGNAL(triggered()), this, SLOT(importTarget()));
}
setFocusPolicy(Qt::NoFocus);
setupUi();
connect(m_project, SIGNAL(addedTarget(ProjectExplorer::Target*)),
this, SLOT(targetAdded(ProjectExplorer::Target*)));
connect(m_project, SIGNAL(removedTarget(ProjectExplorer::Target*)),
this, SLOT(removedTarget(ProjectExplorer::Target*)));
connect(m_project, SIGNAL(activeTargetChanged(ProjectExplorer::Target*)),
this, SLOT(activeTargetChanged(ProjectExplorer::Target*)));
connect(KitManager::instance(), SIGNAL(kitsChanged()),
this, SLOT(updateTargetButtons()));
}
TargetSettingsPanelWidget::~TargetSettingsPanelWidget()
{
delete m_importer;
}
bool TargetSettingsPanelWidget::event(QEvent *event)
{
if (event->type() == QEvent::StatusTip) {
QAction *act = 0;
QMenu *menu = 0;
if (m_addMenu->activeAction()) {
menu = m_addMenu;
act = m_addMenu->activeAction();
} else if (m_changeMenu && m_changeMenu->activeAction()) {
menu = m_changeMenu;
act = m_changeMenu->activeAction();
} else if (m_duplicateMenu && m_duplicateMenu->activeAction()) {
menu = m_duplicateMenu;
act = m_duplicateMenu->activeAction();
} else {
return QWidget::event(event);
}
QStatusTipEvent *ev = static_cast<QStatusTipEvent *>(event);
ev->accept();
if (act != m_lastAction)
QToolTip::showText(QPoint(), QString());
m_lastAction = act;
if (act) {
QRect actionRect = menu->actionGeometry(act);
actionRect.translate(menu->pos());
QPoint p = QCursor::pos();
if (!actionRect.contains(p))
p = actionRect.center();
p.setY(actionRect.center().y());
QToolTip::showText(p, ev->tip(), menu, menu->actionGeometry(act));
} else {
QToolTip::showText(QPoint(), QString());
}
return true;
}
return QWidget::event(event);
}
void TargetSettingsPanelWidget::setupUi()
{
QVBoxLayout *viewLayout = new QVBoxLayout(this);
viewLayout->setMargin(0);
viewLayout->setSpacing(0);
m_selector = new TargetSettingsWidget(this);
viewLayout->addWidget(m_selector);
// Setup our container for the contents:
m_centralWidget = new QStackedWidget(this);
m_selector->setCentralWidget(m_centralWidget);
// no target label:
m_noTargetLabel = new QWidget;
QVBoxLayout *noTargetLayout = new QVBoxLayout(m_noTargetLabel);
noTargetLayout->setMargin(0);
QLabel *label = new QLabel(m_noTargetLabel);
label->setText(tr("No kit defined in this project."));
{
QFont f = label->font();
f.setPointSizeF(f.pointSizeF() * 1.4);
f.setBold(true);
label->setFont(f);
}
label->setMargin(10);
label->setAlignment(Qt::AlignTop);
noTargetLayout->addWidget(label);
noTargetLayout->addStretch(10);
m_centralWidget->addWidget(m_noTargetLabel);
foreach (Target *t, m_project->targets())
targetAdded(t);
// Now set the correct target
int index = m_targets.indexOf(m_project->activeTarget());
m_selector->setCurrentIndex(index);
currentTargetChanged(index, m_selector->currentSubIndex());
connect(m_selector, SIGNAL(currentChanged(int,int)),
this, SLOT(currentTargetChanged(int,int)));
connect(m_selector, SIGNAL(manageButtonClicked()),
this, SLOT(openTargetPreferences()));
connect(m_selector, SIGNAL(toolTipRequested(QPoint,int)),
this, SLOT(showTargetToolTip(QPoint,int)));
connect(m_selector, SIGNAL(menuShown(int)),
this, SLOT(menuShown(int)));
connect(m_addMenu, SIGNAL(triggered(QAction*)),
this, SLOT(addActionTriggered(QAction*)));
m_selector->setAddButtonMenu(m_addMenu);
m_selector->setTargetMenu(m_targetMenu);
updateTargetButtons();
}
void TargetSettingsPanelWidget::currentTargetChanged(int targetIndex, int subIndex)
{
if (targetIndex < -1 || targetIndex >= m_targets.count())
return;
if (subIndex < -1 || subIndex >= 2)
return;
if (targetIndex == -1 || subIndex == -1) { // no more kits!
delete m_panelWidgets[0];
m_panelWidgets[0] = 0;
delete m_panelWidgets[1];
m_panelWidgets[1] = 0;
m_centralWidget->setCurrentWidget(m_noTargetLabel);
return;
}
Target *target = m_targets.at(targetIndex);
// Target was not actually changed:
if (m_currentTarget == target) {
if (m_panelWidgets[subIndex])
m_centralWidget->setCurrentWidget(m_panelWidgets[subIndex]);
else
m_centralWidget->setCurrentWidget(m_noTargetLabel);
return;
}
// Target has changed:
m_currentTarget = target;
PanelsWidget *buildPanel = new PanelsWidget(m_centralWidget);
PanelsWidget *runPanel = new PanelsWidget(m_centralWidget);
foreach (ITargetPanelFactory *panelFactory, ExtensionSystem::PluginManager::getObjects<ITargetPanelFactory>()) {
if (panelFactory->id() == QLatin1String(BUILDSETTINGS_PANEL_ID)) {
PropertiesPanel *panel = panelFactory->createPanel(target);
buildPanel->addPropertiesPanel(panel);
continue;
}
if (panelFactory->id() == QLatin1String(RUNSETTINGS_PANEL_ID)) {
PropertiesPanel *panel = panelFactory->createPanel(target);
runPanel->addPropertiesPanel(panel);
continue;
}
}
m_centralWidget->addWidget(buildPanel);
m_centralWidget->addWidget(runPanel);
m_centralWidget->setCurrentWidget(subIndex == 0 ? buildPanel : runPanel);
delete m_panelWidgets[0];
m_panelWidgets[0] = buildPanel;
delete m_panelWidgets[1];
m_panelWidgets[1] = runPanel;
m_project->setActiveTarget(target);
}
void TargetSettingsPanelWidget::menuShown(int targetIndex)
{
m_menuTargetIndex = targetIndex;
}
void TargetSettingsPanelWidget::changeActionTriggered(QAction *action)
{
Kit *k = KitManager::find(action->data().value<Core::Id>());
Target *sourceTarget = m_targets.at(m_menuTargetIndex);
Target *newTarget = cloneTarget(sourceTarget, k);
if (newTarget) {
m_project->addTarget(newTarget);
m_project->setActiveTarget(newTarget);
m_project->removeTarget(sourceTarget);
}
}
void TargetSettingsPanelWidget::duplicateActionTriggered(QAction *action)
{
Kit *k = KitManager::find(action->data().value<Core::Id>());
Target *newTarget = cloneTarget(m_targets.at(m_menuTargetIndex), k);
if (newTarget) {
m_project->addTarget(newTarget);
m_project->setActiveTarget(newTarget);
}
}
void TargetSettingsPanelWidget::addActionTriggered(QAction *action)
{
const QVariant data = action->data();
if (data.canConvert<Core::Id>()) { // id of kit
Kit *k = KitManager::find(action->data().value<Core::Id>());
QTC_ASSERT(!m_project->target(k), return);
Target *target = m_project->createTarget(k);
if (!target)
return;
m_project->addTarget(target);
} else {
QTC_ASSERT(data.canConvert<IPotentialKit *>(), return);
IPotentialKit *potentialKit = data.value<IPotentialKit *>();
potentialKit->executeFromMenu();
}
}
Target *TargetSettingsPanelWidget::cloneTarget(Target *sourceTarget, Kit *k)
{
Target *newTarget = new Target(m_project, k);
QStringList buildconfigurationError;
QStringList deployconfigurationError;
QStringList runconfigurationError;
foreach (BuildConfiguration *sourceBc, sourceTarget->buildConfigurations()) {
IBuildConfigurationFactory *factory = IBuildConfigurationFactory::find(newTarget, sourceBc);
if (!factory) {
buildconfigurationError << sourceBc->displayName();
continue;
}
BuildConfiguration *newBc = factory->clone(newTarget, sourceBc);
if (!newBc) {
buildconfigurationError << sourceBc->displayName();
continue;
}
newBc->setDisplayName(sourceBc->displayName());
newTarget->addBuildConfiguration(newBc);
if (sourceTarget->activeBuildConfiguration() == sourceBc)
newTarget->setActiveBuildConfiguration(newBc);
}
if (!newTarget->activeBuildConfiguration()) {
QList<BuildConfiguration *> bcs = newTarget->buildConfigurations();
if (!bcs.isEmpty())
newTarget->setActiveBuildConfiguration(bcs.first());
}
foreach (DeployConfiguration *sourceDc, sourceTarget->deployConfigurations()) {
DeployConfigurationFactory *factory = DeployConfigurationFactory::find(newTarget, sourceDc);
if (!factory) {
deployconfigurationError << sourceDc->displayName();
continue;
}
DeployConfiguration *newDc = factory->clone(newTarget, sourceDc);
if (!newDc) {
deployconfigurationError << sourceDc->displayName();
continue;
}
newDc->setDisplayName(sourceDc->displayName());
newTarget->addDeployConfiguration(newDc);
if (sourceTarget->activeDeployConfiguration() == sourceDc)
newTarget->setActiveDeployConfiguration(newDc);
}
if (!newTarget->activeBuildConfiguration()) {
QList<DeployConfiguration *> dcs = newTarget->deployConfigurations();
if (!dcs.isEmpty())
newTarget->setActiveDeployConfiguration(dcs.first());
}
foreach (RunConfiguration *sourceRc, sourceTarget->runConfigurations()) {
IRunConfigurationFactory *factory = IRunConfigurationFactory::find(newTarget, sourceRc);
if (!factory) {
runconfigurationError << sourceRc->displayName();
continue;
}
RunConfiguration *newRc = factory->clone(newTarget, sourceRc);
if (!newRc) {
runconfigurationError << sourceRc->displayName();
continue;
}
newRc->setDisplayName(sourceRc->displayName());
newTarget->addRunConfiguration(newRc);
if (sourceTarget->activeRunConfiguration() == sourceRc)
newTarget->setActiveRunConfiguration(newRc);
}
if (!newTarget->activeRunConfiguration()) {
QList<RunConfiguration *> rcs = newTarget->runConfigurations();
if (!rcs.isEmpty())
newTarget->setActiveRunConfiguration(rcs.first());
}
bool fatalError = false;
if (buildconfigurationError.count() == sourceTarget->buildConfigurations().count())
fatalError = true;
if (deployconfigurationError.count() == sourceTarget->deployConfigurations().count())
fatalError = true;
if (runconfigurationError.count() == sourceTarget->runConfigurations().count())
fatalError = true;
if (fatalError) {
// That could be a more granular error message
QMessageBox::critical(Core::ICore::mainWindow(),
tr("Incompatible Kit"),
tr("Kit %1 is incompatible with kit %2.")
.arg(sourceTarget->kit()->displayName())
.arg(k->displayName()));
delete newTarget;
newTarget = 0;
} else if (!buildconfigurationError.isEmpty()
|| !deployconfigurationError.isEmpty()
|| ! runconfigurationError.isEmpty()) {
QString error;
if (!buildconfigurationError.isEmpty())
error += tr("Build configurations:")
+ QLatin1Char('\n')
+ buildconfigurationError.join(QLatin1String("\n"));
if (!deployconfigurationError.isEmpty()) {
if (!error.isEmpty())
error.append(QLatin1Char('\n'));
error += tr("Deploy configurations:")
+ QLatin1Char('\n')
+ deployconfigurationError.join(QLatin1String("\n"));
}
if (!runconfigurationError.isEmpty()) {
if (!error.isEmpty())
error.append(QLatin1Char('\n'));
error += tr("Run configurations") + QLatin1Char(' ')
+ runconfigurationError.join(QLatin1String("\n"));
}
QMessageBox msgBox(Core::ICore::mainWindow());
msgBox.setIcon(QMessageBox::Warning);
msgBox.setWindowTitle(tr("Partially Incompatible Kit"));
msgBox.setText(tr("Some configurations could not be copied."));
msgBox.setDetailedText(error);
msgBox.setStandardButtons(QMessageBox::Ok | QMessageBox::Cancel);
if (msgBox.exec() != QDialog::Accepted) {
delete newTarget;
newTarget = 0;
}
}
return newTarget;
}
void TargetSettingsPanelWidget::removeTarget()
{
Target *t = m_targets.at(m_menuTargetIndex);
removeTarget(t);
}
void TargetSettingsPanelWidget::removeTarget(Target *t)
{
if (BuildManager::isBuilding(t)) {
QMessageBox box;
QPushButton *closeAnyway = box.addButton(tr("Cancel Build && Remove Kit"), QMessageBox::AcceptRole);
QPushButton *cancelClose = box.addButton(tr("Do Not Remove"), QMessageBox::RejectRole);
box.setDefaultButton(cancelClose);
box.setWindowTitle(tr("Remove Kit %1?").arg(t->displayName()));
box.setText(tr("The kit <b>%1</b> is currently being built.").arg(t->displayName()));
box.setInformativeText(tr("Do you want to cancel the build process and remove the kit anyway?"));
box.exec();
if (box.clickedButton() != closeAnyway)
return;
BuildManager::cancel();
} else {
// We don't show the generic message box on removing the target, if we showed the still building one
int ret = QMessageBox::warning(this, tr("Qt Creator"),
tr("Do you really want to remove the\n"
"\"%1\" kit?").arg(t->displayName()),
QMessageBox::Yes | QMessageBox::No,
QMessageBox::No);
if (ret != QMessageBox::Yes)
return;
}
m_project->removeTarget(t);
}
void TargetSettingsPanelWidget::showTargetToolTip(const QPoint &globalPos, int targetIndex)
{
QTC_ASSERT(targetIndex >= 0 && targetIndex < m_targets.count(), return);
Target *target = m_targets.at(targetIndex);
QToolTip::showText(globalPos, target->kit()->toHtml());
}
void TargetSettingsPanelWidget::targetAdded(ProjectExplorer::Target *target)
{
Q_ASSERT(m_project == target->project());
Q_ASSERT(m_selector);
for (int pos = 0; pos <= m_targets.count(); ++pos) {
if (m_targets.count() == pos ||
m_targets.at(pos)->displayName() > target->displayName()) {
m_targets.insert(pos, target);
m_selector->insertTarget(pos, m_project->hasActiveBuildSettings() ? 0 : 1,
target->displayName());
break;
}
}
connect(target, SIGNAL(displayNameChanged()), this, SLOT(renameTarget()));
updateTargetButtons();
}
void TargetSettingsPanelWidget::removedTarget(ProjectExplorer::Target *target)
{
Q_ASSERT(m_project == target->project());
Q_ASSERT(m_selector);
int index(m_targets.indexOf(target));
if (index < 0)
return;
m_targets.removeAt(index);
m_selector->removeTarget(index);
updateTargetButtons();
}
void TargetSettingsPanelWidget::activeTargetChanged(ProjectExplorer::Target *target)
{
Q_ASSERT(m_selector);
int index = m_targets.indexOf(target);
m_selector->setCurrentIndex(index);
}
namespace {
bool diplayNameSorter(Kit *a, Kit *b)
{
return a->displayName() < b->displayName();
}
}
void TargetSettingsPanelWidget::createAction(Kit *k, QMenu *menu)
{
QAction *action = new QAction(k->displayName(), menu);
action->setData(QVariant::fromValue(k->id()));
QString statusTip = QLatin1String("<html><body>");
QString errorMessage;
if (!m_project->supportsKit(k, &errorMessage)) {
action->setEnabled(false);
statusTip += errorMessage;
}
statusTip += k->toHtml();
action->setStatusTip(statusTip);
menu->addAction(action);
}
void TargetSettingsPanelWidget::updateTargetButtons()
{
if (!m_selector)
return;
m_addMenu->clear();
m_targetMenu->clear();
if (m_importAction)
m_addMenu->addAction(m_importAction);
const QList<IPotentialKit *> potentialKits
= ExtensionSystem::PluginManager::getObjects<IPotentialKit>();
foreach (IPotentialKit *potentialKit, potentialKits) {
if (!potentialKit->isEnabled())
continue;
QAction *action = new QAction(potentialKit->displayName(), m_addMenu);
action->setData(QVariant::fromValue(potentialKit));
m_addMenu->addAction(action);
}
if (!m_addMenu->actions().isEmpty())
m_addMenu->addSeparator();
m_changeMenu = m_targetMenu->addMenu(tr("Change Kit"));
m_duplicateMenu = m_targetMenu->addMenu(tr("Copy to Kit"));
QAction *removeAction = m_targetMenu->addAction(tr("Remove Kit"));
if (m_project->targets().size() < 2)
removeAction->setEnabled(false);
connect(m_changeMenu, SIGNAL(triggered(QAction*)),
this, SLOT(changeActionTriggered(QAction*)));
connect(m_duplicateMenu, SIGNAL(triggered(QAction*)),
this, SLOT(duplicateActionTriggered(QAction*)));
connect(removeAction, SIGNAL(triggered()), this, SLOT(removeTarget()));
QList<Kit *> kits = KitManager::kits();
qSort(kits.begin(), kits.end(), diplayNameSorter);
foreach (Kit *k, kits) {
if (m_project->target(k))
continue;
createAction(k, m_addMenu);
createAction(k, m_changeMenu);
createAction(k, m_duplicateMenu);
}
if (m_changeMenu->actions().isEmpty())
m_changeMenu->setEnabled(false);
if (m_duplicateMenu->actions().isEmpty())
m_duplicateMenu->setEnabled(false);
m_selector->setAddButtonEnabled(!m_addMenu->actions().isEmpty());
}
void TargetSettingsPanelWidget::renameTarget()
{
Target *t = qobject_cast<Target *>(sender());
if (!t)
return;
const int pos = m_targets.indexOf(t);
if (pos < 0)
return;
m_selector->renameTarget(pos, t->displayName());
}
void TargetSettingsPanelWidget::openTargetPreferences()
{
int targetIndex = m_selector->currentIndex();
if (targetIndex >= 0 && targetIndex < m_targets.size()) {
KitOptionsPage *page = ExtensionSystem::PluginManager::getObject<ProjectExplorer::KitOptionsPage>();
if (page)
page->showKit(m_targets.at(targetIndex)->kit());
}
ICore::showOptionsDialog(Constants::PROJECTEXPLORER_SETTINGS_CATEGORY,
Constants::KITS_SETTINGS_PAGE_ID);
}
void TargetSettingsPanelWidget::importTarget()
{
QString toImport = QFileDialog::getExistingDirectory(this, tr("Import directory"), m_project->projectDirectory().toString());
importTarget(Utils::FileName::fromString(toImport));
}
void TargetSettingsPanelWidget::importTarget(const Utils::FileName &path)
{
if (!m_importer)
return;
Target *target = 0;
BuildConfiguration *bc = 0;
QList<BuildInfo *> toImport = m_importer->import(path, false);
foreach (BuildInfo *info, toImport) {
target = m_project->target(info->kitId);
if (!target) {
target = new Target(m_project, KitManager::find(info->kitId));
m_project->addTarget(target);
}
bc = info->factory()->create(target, info);
QTC_ASSERT(bc, continue);
target->addBuildConfiguration(bc);
}
m_project->setActiveTarget(target);
if (target && bc)
target->setActiveBuildConfiguration(bc);
qDeleteAll(toImport);
}
int TargetSettingsPanelWidget::currentSubIndex() const
{
return m_selector->currentSubIndex();
}
void TargetSettingsPanelWidget::setCurrentSubIndex(int subIndex)
{
m_selector->setCurrentSubIndex(subIndex);
}
} // namespace Internal
} // namespace ProjectExplorer