/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2012 Nokia Corporation and/or its subsidiary(-ies). ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** ** GNU Lesser General Public License Usage ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** Other Usage ** ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "makestep.h" #include "cmakeprojectconstants.h" #include "cmakeproject.h" #include "cmaketarget.h" #include "cmakebuildconfiguration.h" #include #include #include #include #include #include #include #include #include #include #include using namespace CMakeProjectManager; using namespace CMakeProjectManager::Internal; using namespace ProjectExplorer; namespace { const char * const MS_ID("CMakeProjectManager.MakeStep"); const char * const CLEAN_KEY("CMakeProjectManager.MakeStep.Clean"); const char * const BUILD_TARGETS_KEY("CMakeProjectManager.MakeStep.BuildTargets"); const char * const ADDITIONAL_ARGUMENTS_KEY("CMakeProjectManager.MakeStep.AdditionalArguments"); } // TODO: Move progress information into an IOutputParser! MakeStep::MakeStep(BuildStepList *bsl) : AbstractProcessStep(bsl, QLatin1String(MS_ID)), m_clean(false), m_futureInterface(0) { ctor(); } MakeStep::MakeStep(BuildStepList *bsl, const QString &id) : AbstractProcessStep(bsl, id), m_clean(false), m_futureInterface(0) { ctor(); } MakeStep::MakeStep(BuildStepList *bsl, MakeStep *bs) : AbstractProcessStep(bsl, bs), m_clean(bs->m_clean), m_futureInterface(0), m_buildTargets(bs->m_buildTargets), m_additionalArguments(Utils::QtcProcess::joinArgs(bs->m_buildTargets)) { ctor(); } void MakeStep::ctor() { m_percentProgress = QRegExp("^\\[\\s*(\\d*)%\\]"); //: Default display name for the cmake make step. setDefaultDisplayName(tr("Make")); } MakeStep::~MakeStep() { } CMakeBuildConfiguration *MakeStep::cmakeBuildConfiguration() const { return static_cast(buildConfiguration()); } void MakeStep::setClean(bool clean) { m_clean = clean; } QVariantMap MakeStep::toMap() const { QVariantMap map(AbstractProcessStep::toMap()); map.insert(QLatin1String(CLEAN_KEY), m_clean); map.insert(QLatin1String(BUILD_TARGETS_KEY), m_buildTargets); map.insert(QLatin1String(ADDITIONAL_ARGUMENTS_KEY), m_additionalArguments); return map; } bool MakeStep::fromMap(const QVariantMap &map) { m_clean = map.value(QLatin1String(CLEAN_KEY)).toBool(); m_buildTargets = map.value(QLatin1String(BUILD_TARGETS_KEY)).toStringList(); m_additionalArguments = map.value(QLatin1String(ADDITIONAL_ARGUMENTS_KEY)).toString(); return BuildStep::fromMap(map); } bool MakeStep::init() { CMakeBuildConfiguration *bc = cmakeBuildConfiguration(); QString arguments = Utils::QtcProcess::joinArgs(m_buildTargets); Utils::QtcProcess::addArgs(&arguments, additionalArguments()); setEnabled(true); setIgnoreReturnValue(m_clean); ProcessParameters *pp = processParameters(); pp->setMacroExpander(bc->macroExpander()); pp->setEnvironment(bc->environment()); pp->setWorkingDirectory(bc->buildDirectory()); if (bc->toolChain()) pp->setCommand(bc->toolChain()->makeCommand()); else pp->setCommand(QLatin1String("make")); pp->setArguments(arguments); setOutputParser(new ProjectExplorer::GnuMakeParser()); if (bc->toolChain()) appendOutputParser(bc->toolChain()->outputParser()); outputParser()->setWorkingDirectory(pp->effectiveWorkingDirectory()); return AbstractProcessStep::init(); } void MakeStep::run(QFutureInterface &fi) { m_futureInterface = &fi; m_futureInterface->setProgressRange(0, 100); AbstractProcessStep::run(fi); m_futureInterface->setProgressValue(100); m_futureInterface->reportFinished(); m_futureInterface = 0; } BuildStepConfigWidget *MakeStep::createConfigWidget() { return new MakeStepConfigWidget(this); } bool MakeStep::immutable() const { return false; } void MakeStep::stdOutput(const QString &line) { if (m_percentProgress.indexIn(line) != -1) { bool ok = false; int percent = m_percentProgress.cap(1).toInt(&ok);; if (ok) m_futureInterface->setProgressValue(percent); } AbstractProcessStep::stdOutput(line); } QStringList MakeStep::buildTargets() const { return m_buildTargets; } bool MakeStep::buildsBuildTarget(const QString &target) const { return m_buildTargets.contains(target); } void MakeStep::setBuildTarget(const QString &buildTarget, bool on) { QStringList old = m_buildTargets; if (on && !old.contains(buildTarget)) old << buildTarget; else if (!on && old.contains(buildTarget)) old.removeOne(buildTarget); m_buildTargets = old; } void MakeStep::setBuildTargets(const QStringList &targets) { m_buildTargets = targets; } void MakeStep::clearBuildTargets() { m_buildTargets.clear(); } QString MakeStep::additionalArguments() const { return m_additionalArguments; } void MakeStep::setAdditionalArguments(const QString &list) { m_additionalArguments = list; } // // MakeStepConfigWidget // MakeStepConfigWidget::MakeStepConfigWidget(MakeStep *makeStep) : m_makeStep(makeStep) { QFormLayout *fl = new QFormLayout(this); fl->setMargin(0); fl->setFieldGrowthPolicy(QFormLayout::ExpandingFieldsGrow); setLayout(fl); m_additionalArguments = new QLineEdit(this); fl->addRow(tr("Additional arguments:"), m_additionalArguments); m_additionalArguments->setText(m_makeStep->additionalArguments()); m_buildTargetsList = new QListWidget; m_buildTargetsList->setMinimumHeight(200); fl->addRow(tr("Targets:"), m_buildTargetsList); // TODO update this list also on rescans of the CMakeLists.txt // TODO shouldn't be accessing project CMakeProject *pro = m_makeStep->cmakeBuildConfiguration()->cmakeTarget()->cmakeProject(); foreach (const QString& buildTarget, pro->buildTargetTitles()) { QListWidgetItem *item = new QListWidgetItem(buildTarget, m_buildTargetsList); item->setFlags(item->flags() | Qt::ItemIsUserCheckable); item->setCheckState(m_makeStep->buildsBuildTarget(item->text()) ? Qt::Checked : Qt::Unchecked); } updateDetails(); connect(m_additionalArguments, SIGNAL(textEdited(const QString &)), this, SLOT(additionalArgumentsEdited())); connect(m_buildTargetsList, SIGNAL(itemChanged(QListWidgetItem*)), this, SLOT(itemChanged(QListWidgetItem*))); connect(ProjectExplorer::ProjectExplorerPlugin::instance(), SIGNAL(settingsChanged()), this, SLOT(updateDetails())); connect(pro, SIGNAL(buildTargetsChanged()), this, SLOT(buildTargetsChanged())); } void MakeStepConfigWidget::additionalArgumentsEdited() { m_makeStep->setAdditionalArguments(m_additionalArguments->text()); updateDetails(); } void MakeStepConfigWidget::itemChanged(QListWidgetItem *item) { m_makeStep->setBuildTarget(item->text(), item->checkState() & Qt::Checked); updateDetails(); } QString MakeStepConfigWidget::displayName() const { return tr("Make", "CMakeProjectManager::MakeStepConfigWidget display name."); } void MakeStepConfigWidget::buildTargetsChanged() { disconnect(m_buildTargetsList, SIGNAL(itemChanged(QListWidgetItem*)), this, SLOT(itemChanged(QListWidgetItem*))); m_buildTargetsList->clear(); CMakeProject *pro = m_makeStep->cmakeBuildConfiguration()->cmakeTarget()->cmakeProject(); foreach (const QString& buildTarget, pro->buildTargetTitles()) { QListWidgetItem *item = new QListWidgetItem(buildTarget, m_buildTargetsList); item->setFlags(item->flags() | Qt::ItemIsUserCheckable); item->setCheckState(m_makeStep->buildsBuildTarget(item->text()) ? Qt::Checked : Qt::Unchecked); } connect(m_buildTargetsList, SIGNAL(itemChanged(QListWidgetItem*)), this, SLOT(itemChanged(QListWidgetItem*))); updateSummary(); } void MakeStepConfigWidget::updateDetails() { CMakeBuildConfiguration *bc = m_makeStep->cmakeBuildConfiguration(); ProjectExplorer::ToolChain *tc = bc->toolChain(); if (tc) { QString arguments = Utils::QtcProcess::joinArgs(m_makeStep->m_buildTargets); Utils::QtcProcess::addArgs(&arguments, m_makeStep->additionalArguments()); ProcessParameters param; param.setMacroExpander(bc->macroExpander()); param.setEnvironment(bc->environment()); param.setWorkingDirectory(bc->buildDirectory()); param.setCommand(tc->makeCommand()); param.setArguments(arguments); m_summaryText = param.summary(displayName()); } else { m_summaryText = tr("Unknown tool chain"); } emit updateSummary(); } QString MakeStepConfigWidget::summaryText() const { return m_summaryText; } // // MakeStepFactory // MakeStepFactory::MakeStepFactory(QObject *parent) : ProjectExplorer::IBuildStepFactory(parent) { } MakeStepFactory::~MakeStepFactory() { } bool MakeStepFactory::canCreate(BuildStepList *parent, const QString &id) const { if (parent->target()->project()->id() != QLatin1String(Constants::CMAKEPROJECT_ID)) return false; return QLatin1String(MS_ID) == id; } BuildStep *MakeStepFactory::create(BuildStepList *parent, const QString &id) { if (!canCreate(parent, id)) return 0; MakeStep *step = new MakeStep(parent); if (parent->id() == ProjectExplorer::Constants::BUILDSTEPS_CLEAN) { step->setClean(true); step->setAdditionalArguments("clean"); } return step; } bool MakeStepFactory::canClone(BuildStepList *parent, BuildStep *source) const { return canCreate(parent, source->id()); } BuildStep *MakeStepFactory::clone(BuildStepList *parent, BuildStep *source) { if (!canClone(parent, source)) return 0; return new MakeStep(parent, static_cast(source)); } bool MakeStepFactory::canRestore(BuildStepList *parent, const QVariantMap &map) const { QString id(ProjectExplorer::idFromMap(map)); return canCreate(parent, id); } BuildStep *MakeStepFactory::restore(BuildStepList *parent, const QVariantMap &map) { if (!canRestore(parent, map)) return 0; MakeStep *bs(new MakeStep(parent)); if (bs->fromMap(map)) return bs; delete bs; return 0; } QStringList MakeStepFactory::availableCreationIds(ProjectExplorer::BuildStepList *parent) const { if (parent->target()->project()->id() == QLatin1String(Constants::CMAKEPROJECT_ID)) return QStringList() << QLatin1String(MS_ID); return QStringList(); } QString MakeStepFactory::displayNameForId(const QString &id) const { if (id == QLatin1String(MS_ID)) return tr("Make", "Display name for CMakeProjectManager::MakeStep id."); return QString(); }