2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
** Contact: Qt Software Information (qt-info@nokia.com)
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and Nokia.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
|
|
|
** contact the sales department at qt-sales@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "cmakeproject.h"
|
|
|
|
#include "cmakeprojectconstants.h"
|
|
|
|
#include "cmakeprojectnodes.h"
|
2008-12-09 15:25:01 +01:00
|
|
|
#include "cmakerunconfiguration.h"
|
2008-12-05 16:30:26 +01:00
|
|
|
#include "makestep.h"
|
2009-03-09 18:13:19 +01:00
|
|
|
#include "cmakeopenprojectwizard.h"
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2009-02-16 13:12:12 +01:00
|
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <cpptools/cppmodelmanagerinterface.h>
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <extensionsystem/pluginmanager.h>
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <coreplugin/icore.h>
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtCore/QDebug>
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <QtCore/QDir>
|
|
|
|
#include <QtCore/QProcess>
|
2009-01-12 15:10:33 +01:00
|
|
|
#include <QtGui/QFormLayout>
|
2009-01-23 16:57:38 +01:00
|
|
|
#include <QtGui/QMainWindow>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
using namespace CMakeProjectManager;
|
|
|
|
using namespace CMakeProjectManager::Internal;
|
|
|
|
|
2008-12-12 17:22:02 +01:00
|
|
|
|
|
|
|
// QtCreator CMake Generator wishlist:
|
|
|
|
// Which make targets we need to build to get all executables
|
|
|
|
// What is the make we need to call
|
|
|
|
// What is the actual compiler executable
|
|
|
|
// DEFINES
|
|
|
|
|
|
|
|
// Open Questions
|
|
|
|
// Who sets up the environment for cl.exe ? INCLUDEPATH and so on
|
|
|
|
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
CMakeProject::CMakeProject(CMakeManager *manager, const QString &fileName)
|
2009-02-11 12:14:51 +01:00
|
|
|
: m_manager(manager),
|
|
|
|
m_fileName(fileName),
|
|
|
|
m_rootNode(new CMakeProjectNode(m_fileName)),
|
|
|
|
m_toolChain(0)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
m_file = new CMakeFile(this, fileName);
|
2009-01-12 15:10:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CMakeProject::~CMakeProject()
|
|
|
|
{
|
|
|
|
delete m_rootNode;
|
2009-02-11 12:14:51 +01:00
|
|
|
delete m_toolChain;
|
2009-01-12 15:10:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO also call this method if the CMakeLists.txt file changed, which is also called if the CMakeList.txt is updated
|
|
|
|
// TODO make this function work even if it is reparsing
|
2009-01-23 16:57:38 +01:00
|
|
|
void CMakeProject::parseCMakeLists()
|
2009-01-12 15:10:33 +01:00
|
|
|
{
|
2009-02-11 12:14:51 +01:00
|
|
|
ProjectExplorer::ToolChain *newToolChain = 0;
|
2009-01-23 16:57:38 +01:00
|
|
|
QString sourceDirectory = QFileInfo(m_fileName).absolutePath();
|
2008-12-02 17:56:21 +01:00
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
QString cbpFile = CMakeManager::findCbpFile(buildDirectory(activeBuildConfiguration()));
|
2008-12-02 17:56:21 +01:00
|
|
|
CMakeCbpParser cbpparser;
|
2009-01-12 15:10:33 +01:00
|
|
|
qDebug()<<"Parsing file "<<cbpFile;
|
2008-12-02 17:56:21 +01:00
|
|
|
if (cbpparser.parseCbpFile(cbpFile)) {
|
2009-02-11 12:14:51 +01:00
|
|
|
qDebug()<<"CodeBlocks Compilername"<<cbpparser.compilerName();
|
|
|
|
if (cbpparser.compilerName() == "gcc") {
|
|
|
|
newToolChain = ProjectExplorer::ToolChain::createGccToolChain("gcc");
|
|
|
|
} else if (cbpparser.compilerName() == "msvc8") {
|
|
|
|
// TODO hmm
|
|
|
|
//newToolChain = ProjectExplorer::ToolChain::createMSVCToolChain("//TODO");
|
|
|
|
Q_ASSERT(false);
|
|
|
|
} else {
|
|
|
|
// TODO hmm?
|
|
|
|
}
|
2009-02-16 13:12:12 +01:00
|
|
|
if (ProjectExplorer::ToolChain::equals(newToolChain, m_toolChain)) {
|
2009-02-11 12:14:51 +01:00
|
|
|
delete newToolChain;
|
|
|
|
newToolChain = 0;
|
|
|
|
} else {
|
|
|
|
delete m_toolChain;
|
|
|
|
m_toolChain = newToolChain;
|
|
|
|
}
|
|
|
|
|
2009-01-15 14:48:28 +00:00
|
|
|
m_projectName = cbpparser.projectName();
|
2009-01-12 15:10:33 +01:00
|
|
|
qDebug()<<"Building Tree";
|
2008-12-09 14:13:29 +01:00
|
|
|
// TODO do a intelligent updating of the tree
|
2009-03-03 17:56:03 +01:00
|
|
|
|
|
|
|
QList<ProjectExplorer::FileNode *> fileList = cbpparser.fileList();
|
|
|
|
// Manually add the CMakeLists.txt file
|
|
|
|
fileList.append(new ProjectExplorer::FileNode(sourceDirectory + "/CMakeLists.txt", ProjectExplorer::ProjectFileType, false));
|
|
|
|
buildTree(m_rootNode, fileList);
|
|
|
|
foreach (ProjectExplorer::FileNode *fn, fileList)
|
2008-12-02 12:01:29 +01:00
|
|
|
m_files.append(fn->path());
|
|
|
|
m_files.sort();
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
qDebug()<<"Adding Targets";
|
2008-12-09 14:13:29 +01:00
|
|
|
m_targets = cbpparser.targets();
|
|
|
|
qDebug()<<"Printing targets";
|
|
|
|
foreach(CMakeTarget ct, m_targets) {
|
|
|
|
qDebug()<<ct.title<<" with executable:"<<ct.executable;
|
|
|
|
qDebug()<<"WD:"<<ct.workingDirectory;
|
|
|
|
qDebug()<<ct.makeCommand<<ct.makeCleanCommand;
|
|
|
|
qDebug()<<"";
|
|
|
|
}
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
qDebug()<<"Updating CodeModel";
|
2009-02-11 12:14:51 +01:00
|
|
|
|
|
|
|
QStringList allIncludePaths;
|
|
|
|
QStringList allFrameworkPaths;
|
|
|
|
QList<ProjectExplorer::HeaderPath> allHeaderPaths = m_toolChain->systemHeaderPaths();
|
|
|
|
foreach (ProjectExplorer::HeaderPath headerPath, allHeaderPaths) {
|
|
|
|
if (headerPath.kind() == ProjectExplorer::HeaderPath::FrameworkHeaderPath)
|
|
|
|
allFrameworkPaths.append(headerPath.path());
|
|
|
|
else
|
|
|
|
allIncludePaths.append(headerPath.path());
|
|
|
|
}
|
|
|
|
allIncludePaths.append(cbpparser.includeFiles());
|
2008-12-02 12:01:29 +01:00
|
|
|
CppTools::CppModelManagerInterface *modelmanager = ExtensionSystem::PluginManager::instance()->getObject<CppTools::CppModelManagerInterface>();
|
|
|
|
if (modelmanager) {
|
2008-12-08 10:44:56 +01:00
|
|
|
CppTools::CppModelManagerInterface::ProjectInfo pinfo = modelmanager->projectInfo(this);
|
2009-02-11 12:14:51 +01:00
|
|
|
pinfo.includePaths = allIncludePaths;
|
2008-12-02 17:56:21 +01:00
|
|
|
// TODO we only want C++ files, not all other stuff that might be in the project
|
2008-12-08 10:44:56 +01:00
|
|
|
pinfo.sourceFiles = m_files;
|
2009-02-11 12:14:51 +01:00
|
|
|
pinfo.defines = m_toolChain->predefinedMacros();
|
|
|
|
pinfo.frameworkPaths = allFrameworkPaths;
|
2008-12-08 10:44:56 +01:00
|
|
|
modelmanager->updateProjectInfo(pinfo);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// TODO report error
|
2009-02-11 12:14:51 +01:00
|
|
|
delete m_toolChain;
|
|
|
|
m_toolChain = 0;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-16 13:12:12 +01:00
|
|
|
QString CMakeProject::buildParser(const QString &buildConfiguration) const
|
|
|
|
{
|
2009-03-09 18:13:19 +01:00
|
|
|
// TODO this is actually slightly wrong, but do i care?
|
|
|
|
// this should call toolchain(buildConfiguration)
|
2009-02-16 13:12:12 +01:00
|
|
|
if (!m_toolChain)
|
|
|
|
return QString::null;
|
|
|
|
if (m_toolChain->type() == ProjectExplorer::ToolChain::GCC
|
|
|
|
|| m_toolChain->type() == ProjectExplorer::ToolChain::LinuxICC
|
|
|
|
|| m_toolChain->type() == ProjectExplorer::ToolChain::MinGW) {
|
|
|
|
return ProjectExplorer::Constants::BUILD_PARSER_GCC;
|
|
|
|
} else if (m_toolChain->type() == ProjectExplorer::ToolChain::MSVC
|
|
|
|
|| m_toolChain->type() == ProjectExplorer::ToolChain::WINCE) {
|
|
|
|
return ProjectExplorer::Constants::BUILD_PARSER_MSVC;
|
|
|
|
}
|
|
|
|
return QString::null;
|
|
|
|
}
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
QStringList CMakeProject::targets() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
QStringList results;
|
|
|
|
foreach(const CMakeTarget &ct, m_targets)
|
|
|
|
results << ct.title;
|
|
|
|
return results;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeProject::buildTree(CMakeProjectNode *rootNode, QList<ProjectExplorer::FileNode *> list)
|
|
|
|
{
|
|
|
|
//m_rootNode->addFileNodes(fileList, m_rootNode);
|
|
|
|
qSort(list.begin(), list.end(), ProjectExplorer::ProjectNode::sortNodesByPath);
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (ProjectExplorer::FileNode *fn, list) {
|
2008-12-02 12:01:29 +01:00
|
|
|
// Get relative path to rootNode
|
|
|
|
QString parentDir = QFileInfo(fn->path()).absolutePath();
|
|
|
|
ProjectExplorer::FolderNode *folder = findOrCreateFolder(rootNode, parentDir);
|
|
|
|
rootNode->addFileNodes(QList<ProjectExplorer::FileNode *>()<< fn, folder);
|
|
|
|
}
|
|
|
|
//m_rootNode->addFileNodes(list, rootNode);
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::FolderNode *CMakeProject::findOrCreateFolder(CMakeProjectNode *rootNode, QString directory)
|
|
|
|
{
|
|
|
|
QString relativePath = QDir(QFileInfo(rootNode->path()).path()).relativeFilePath(directory);
|
2009-03-03 17:56:03 +01:00
|
|
|
QStringList parts = relativePath.split("/", QString::SkipEmptyParts);
|
2008-12-02 12:01:29 +01:00
|
|
|
ProjectExplorer::FolderNode *parent = rootNode;
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (const QString &part, parts) {
|
2008-12-02 12:01:29 +01:00
|
|
|
// Find folder in subFolders
|
|
|
|
bool found = false;
|
2008-12-09 11:07:24 +01:00
|
|
|
foreach (ProjectExplorer::FolderNode *folder, parent->subFolderNodes()) {
|
2008-12-02 12:01:29 +01:00
|
|
|
if (QFileInfo(folder->path()).fileName() == part) {
|
|
|
|
// yeah found something :)
|
|
|
|
parent = folder;
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
// No FolderNode yet, so create it
|
|
|
|
ProjectExplorer::FolderNode *tmp = new ProjectExplorer::FolderNode(part);
|
|
|
|
rootNode->addFolderNodes(QList<ProjectExplorer::FolderNode *>() << tmp, parent);
|
|
|
|
parent = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeProject::name() const
|
|
|
|
{
|
2009-01-15 14:48:28 +00:00
|
|
|
return m_projectName;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2009-02-11 12:14:51 +01:00
|
|
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
Core::IFile *CMakeProject::file() const
|
|
|
|
{
|
|
|
|
return m_file;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::IProjectManager *CMakeProject::projectManager() const
|
|
|
|
{
|
|
|
|
return m_manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<ProjectExplorer::Project *> CMakeProject::dependsOn()
|
|
|
|
{
|
|
|
|
return QList<Project *>();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMakeProject::isApplication() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::Environment CMakeProject::environment(const QString &buildConfiguration) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(buildConfiguration)
|
|
|
|
//TODO
|
|
|
|
return ProjectExplorer::Environment::systemEnvironment();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeProject::buildDirectory(const QString &buildConfiguration) const
|
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
QString buildDirectory = value(buildConfiguration, "buildDirectory").toString();
|
|
|
|
if (buildDirectory.isEmpty())
|
2009-01-15 19:29:42 +00:00
|
|
|
buildDirectory = QFileInfo(m_fileName).absolutePath() + "/qtcreator-build";
|
2009-01-12 15:10:33 +01:00
|
|
|
return buildDirectory;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::BuildStepConfigWidget *CMakeProject::createConfigWidget()
|
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
return new CMakeBuildSettingsWidget(this);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QList<ProjectExplorer::BuildStepConfigWidget*> CMakeProject::subConfigWidgets()
|
|
|
|
{
|
|
|
|
return QList<ProjectExplorer::BuildStepConfigWidget*>();
|
|
|
|
}
|
|
|
|
|
|
|
|
// This method is called for new build configurations
|
|
|
|
// You should probably set some default values in this method
|
|
|
|
void CMakeProject::newBuildConfiguration(const QString &buildConfiguration)
|
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
// Default to all
|
|
|
|
makeStep()->setBuildTarget(buildConfiguration, "all", true);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ProjectExplorer::ProjectNode *CMakeProject::rootProjectNode() const
|
|
|
|
{
|
|
|
|
return m_rootNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QStringList CMakeProject::files(FilesMode fileMode) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(fileMode);
|
|
|
|
// TODO
|
|
|
|
return m_files;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeProject::saveSettingsImpl(ProjectExplorer::PersistentSettingsWriter &writer)
|
|
|
|
{
|
|
|
|
// TODO
|
2008-12-09 14:13:29 +01:00
|
|
|
Project::saveSettingsImpl(writer);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
MakeStep *CMakeProject::makeStep() const
|
|
|
|
{
|
|
|
|
foreach (ProjectExplorer::BuildStep *bs, buildSteps()) {
|
|
|
|
MakeStep *ms = qobject_cast<MakeStep *>(bs);
|
|
|
|
if (ms)
|
|
|
|
return ms;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void CMakeProject::restoreSettingsImpl(ProjectExplorer::PersistentSettingsReader &reader)
|
|
|
|
{
|
2008-12-09 14:13:29 +01:00
|
|
|
Project::restoreSettingsImpl(reader);
|
2009-01-23 16:57:38 +01:00
|
|
|
bool hasUserFile = !buildConfigurations().isEmpty();
|
|
|
|
if (!hasUserFile) {
|
|
|
|
// Ask the user for where he wants to build it
|
|
|
|
// and the cmake command line
|
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
CMakeOpenProjectWizard copw(m_manager, QFileInfo(m_fileName).absolutePath());
|
|
|
|
copw.exec();
|
|
|
|
// TODO handle cancel....
|
2009-01-23 16:57:38 +01:00
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
qDebug()<<"ccd.buildDirectory()"<<copw.buildDirectory();
|
2009-01-23 16:57:38 +01:00
|
|
|
|
|
|
|
// Now create a standard build configuration
|
2008-12-05 16:30:26 +01:00
|
|
|
MakeStep *makeStep = new MakeStep(this);
|
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
insertBuildStep(0, makeStep);
|
2008-12-05 16:30:26 +01:00
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
addBuildConfiguration("all");
|
|
|
|
setActiveBuildConfiguration("all");
|
|
|
|
makeStep->setBuildTarget("all", "all", true);
|
2009-03-09 18:13:19 +01:00
|
|
|
if (!copw.buildDirectory().isEmpty())
|
|
|
|
setValue("all", "buildDirectory", copw.buildDirectory());
|
|
|
|
//TODO save arguments somewhere copw.arguments()
|
|
|
|
} else {
|
|
|
|
// We have a user file, but we could still be missing the cbp file
|
|
|
|
// TODO check that we have a cbp file and if not, open up a dialog ?
|
|
|
|
// or simply run createXml with the saved settings
|
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
}
|
|
|
|
|
2009-03-09 18:13:19 +01:00
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
parseCMakeLists(); // Gets the directory from the active buildconfiguration
|
2008-12-12 17:22:02 +01:00
|
|
|
|
2009-01-23 16:57:38 +01:00
|
|
|
if (!hasUserFile) {
|
2009-01-12 15:10:33 +01:00
|
|
|
// Create run configurations for m_targets
|
|
|
|
qDebug()<<"Create run configurations of m_targets";
|
2008-12-12 17:22:02 +01:00
|
|
|
bool setActive = false;
|
2008-12-09 14:13:29 +01:00
|
|
|
foreach(const CMakeTarget &ct, m_targets) {
|
2009-01-12 17:51:50 +01:00
|
|
|
if (ct.executable.isEmpty())
|
|
|
|
continue;
|
2008-12-09 14:13:29 +01:00
|
|
|
QSharedPointer<ProjectExplorer::RunConfiguration> rc(new CMakeRunConfiguration(this, ct.executable, ct.workingDirectory));
|
|
|
|
addRunConfiguration(rc);
|
2008-12-12 17:22:02 +01:00
|
|
|
// The first one gets the honour of beeing the active one
|
|
|
|
if (!setActive) {
|
|
|
|
setActiveRunConfiguration(rc);
|
|
|
|
setActive = true;
|
|
|
|
}
|
2008-12-09 14:13:29 +01:00
|
|
|
}
|
|
|
|
|
2008-12-05 16:30:26 +01:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CMakeFile::CMakeFile(CMakeProject *parent, QString fileName)
|
|
|
|
: Core::IFile(parent), m_project(parent), m_fileName(fileName)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMakeFile::save(const QString &fileName)
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
// Once we have an texteditor open for this file, we probably do
|
|
|
|
// need to implement this, don't we.
|
|
|
|
Q_UNUSED(fileName);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeFile::fileName() const
|
|
|
|
{
|
|
|
|
return m_fileName;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeFile::defaultPath() const
|
|
|
|
{
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeFile::suggestedFileName() const
|
|
|
|
{
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeFile::mimeType() const
|
|
|
|
{
|
|
|
|
return Constants::CMAKEMIMETYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool CMakeFile::isModified() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMakeFile::isReadOnly() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CMakeFile::isSaveAsAllowed() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeFile::modified(ReloadBehavior *behavior)
|
|
|
|
{
|
|
|
|
Q_UNUSED(behavior);
|
|
|
|
}
|
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
CMakeBuildSettingsWidget::CMakeBuildSettingsWidget(CMakeProject *project)
|
|
|
|
: m_project(project)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
QFormLayout *fl = new QFormLayout(this);
|
|
|
|
setLayout(fl);
|
|
|
|
m_pathChooser = new Core::Utils::PathChooser(this);
|
|
|
|
m_pathChooser->setEnabled(false);
|
|
|
|
// TODO currently doesn't work
|
|
|
|
// since creating the cbp file also creates makefiles
|
|
|
|
// and then cmake builds in that directory instead of shadow building
|
|
|
|
// We need our own generator for that to work
|
|
|
|
connect(m_pathChooser, SIGNAL(changed()), this, SLOT(buildDirectoryChanged()));
|
|
|
|
fl->addRow("Build directory:", m_pathChooser);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
QString CMakeBuildSettingsWidget::displayName() const
|
|
|
|
{
|
|
|
|
return "CMake";
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeBuildSettingsWidget::init(const QString &buildConfiguration)
|
|
|
|
{
|
2009-01-12 15:10:33 +01:00
|
|
|
m_buildConfiguration = buildConfiguration;
|
|
|
|
m_pathChooser->setPath(m_project->buildDirectory(buildConfiguration));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2008-12-02 17:56:21 +01:00
|
|
|
|
2009-01-12 15:10:33 +01:00
|
|
|
void CMakeBuildSettingsWidget::buildDirectoryChanged()
|
|
|
|
{
|
|
|
|
m_project->setValue(m_buildConfiguration, "buildDirectory", m_pathChooser->path());
|
|
|
|
}
|
|
|
|
|
|
|
|
/////
|
|
|
|
// CMakeCbpParser
|
|
|
|
////
|
|
|
|
|
2008-12-02 17:56:21 +01:00
|
|
|
bool CMakeCbpParser::parseCbpFile(const QString &fileName)
|
|
|
|
{
|
|
|
|
QFile fi(fileName);
|
|
|
|
if (fi.exists() && fi.open(QFile::ReadOnly)) {
|
|
|
|
setDevice(&fi);
|
|
|
|
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (name() == "CodeBlocks_project_file") {
|
|
|
|
parseCodeBlocks_project_file();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fi.close();
|
|
|
|
m_includeFiles.sort();
|
|
|
|
m_includeFiles.removeDuplicates();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseCodeBlocks_project_file()
|
|
|
|
{
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (name() == "Project") {
|
|
|
|
parseProject();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseProject()
|
|
|
|
{
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
2009-01-15 14:48:28 +00:00
|
|
|
} else if (name() == "Option") {
|
|
|
|
parseOption();
|
2008-12-02 17:56:21 +01:00
|
|
|
} else if (name() == "Unit") {
|
|
|
|
parseUnit();
|
|
|
|
} else if (name() == "Build") {
|
|
|
|
parseBuild();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseBuild()
|
|
|
|
{
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (name() == "Target") {
|
|
|
|
parseTarget();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseTarget()
|
|
|
|
{
|
2008-12-08 12:24:31 +01:00
|
|
|
m_targetType = false;
|
2008-12-09 14:13:29 +01:00
|
|
|
m_target.clear();
|
|
|
|
|
|
|
|
if (attributes().hasAttribute("title"))
|
|
|
|
m_target.title = attributes().value("title").toString();
|
2008-12-09 14:33:09 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
2009-01-12 17:51:50 +01:00
|
|
|
if (m_targetType || m_target.title == "all" || m_target.title == "install") {
|
2008-12-09 14:13:29 +01:00
|
|
|
m_targets.append(m_target);
|
2008-12-08 12:24:31 +01:00
|
|
|
}
|
2008-12-02 17:56:21 +01:00
|
|
|
return;
|
|
|
|
} else if (name() == "Compiler") {
|
|
|
|
parseCompiler();
|
2008-12-08 12:24:31 +01:00
|
|
|
} else if (name() == "Option") {
|
|
|
|
parseTargetOption();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseTargetOption()
|
|
|
|
{
|
|
|
|
if (attributes().hasAttribute("output"))
|
2008-12-09 14:13:29 +01:00
|
|
|
m_target.executable = attributes().value("output").toString();
|
2009-01-16 13:47:40 +01:00
|
|
|
else if (attributes().hasAttribute("type") && (attributes().value("type") == "1" || attributes().value("type") == "0"))
|
2008-12-08 12:24:31 +01:00
|
|
|
m_targetType = true;
|
2008-12-09 14:13:29 +01:00
|
|
|
else if (attributes().hasAttribute("working_dir"))
|
|
|
|
m_target.workingDirectory = attributes().value("working_dir").toString();
|
2008-12-09 14:33:09 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-09 14:13:29 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (name() == "MakeCommand") {
|
|
|
|
parseMakeCommand();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-01-15 14:48:28 +00:00
|
|
|
QString CMakeCbpParser::projectName() const
|
|
|
|
{
|
|
|
|
return m_projectName;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseOption()
|
|
|
|
{
|
|
|
|
if (attributes().hasAttribute("title"))
|
|
|
|
m_projectName = attributes().value("title").toString();
|
2009-01-15 16:06:37 +01:00
|
|
|
|
2009-02-11 12:14:51 +01:00
|
|
|
if (attributes().hasAttribute("compiler"))
|
|
|
|
m_compiler = attributes().value("compiler").toString();
|
|
|
|
|
2009-01-15 16:06:37 +01:00
|
|
|
while (!atEnd()) {
|
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if(isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
2009-01-15 14:48:28 +00:00
|
|
|
}
|
|
|
|
|
2008-12-09 14:13:29 +01:00
|
|
|
void CMakeCbpParser::parseMakeCommand()
|
|
|
|
{
|
2008-12-09 14:33:09 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-09 14:13:29 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (name() == "Build") {
|
|
|
|
parseTargetBuild();
|
|
|
|
} else if (name() == "Clean") {
|
|
|
|
parseTargetClean();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseTargetBuild()
|
|
|
|
{
|
|
|
|
if (attributes().hasAttribute("command"))
|
|
|
|
m_target.makeCommand = attributes().value("command").toString();
|
2008-12-09 14:33:09 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-09 14:13:29 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseTargetClean()
|
|
|
|
{
|
|
|
|
if (attributes().hasAttribute("command"))
|
|
|
|
m_target.makeCleanCommand = attributes().value("command").toString();
|
2008-12-09 14:33:09 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-08 12:24:31 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
2008-12-02 17:56:21 +01:00
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseCompiler()
|
|
|
|
{
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (name() == "Add") {
|
|
|
|
parseAdd();
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseAdd()
|
|
|
|
{
|
|
|
|
m_includeFiles.append(attributes().value("directory").toString());
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseUnit()
|
|
|
|
{
|
|
|
|
//qDebug()<<stream.attributes().value("filename");
|
|
|
|
QString fileName = attributes().value("filename").toString();
|
|
|
|
if (!fileName.endsWith(".rule"))
|
|
|
|
m_fileList.append( new ProjectExplorer::FileNode(fileName, ProjectExplorer::SourceType, false));
|
2008-12-09 11:07:24 +01:00
|
|
|
while (!atEnd()) {
|
2008-12-02 17:56:21 +01:00
|
|
|
readNext();
|
|
|
|
if (isEndElement()) {
|
|
|
|
return;
|
|
|
|
} else if (isStartElement()) {
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMakeCbpParser::parseUnknownElement()
|
|
|
|
{
|
2008-12-17 15:51:48 +01:00
|
|
|
Q_ASSERT(isStartElement());
|
2008-12-02 17:56:21 +01:00
|
|
|
|
|
|
|
while (!atEnd()) {
|
|
|
|
readNext();
|
|
|
|
|
|
|
|
if (isEndElement())
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (isStartElement())
|
|
|
|
parseUnknownElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<ProjectExplorer::FileNode *> CMakeCbpParser::fileList()
|
|
|
|
{
|
|
|
|
return m_fileList;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList CMakeCbpParser::includeFiles()
|
|
|
|
{
|
|
|
|
return m_includeFiles;
|
|
|
|
}
|
2008-12-09 14:13:29 +01:00
|
|
|
|
|
|
|
QList<CMakeTarget> CMakeCbpParser::targets()
|
|
|
|
{
|
|
|
|
return m_targets;
|
|
|
|
}
|
|
|
|
|
2009-02-11 12:14:51 +01:00
|
|
|
QString CMakeCbpParser::compilerName() const
|
|
|
|
{
|
|
|
|
return m_compiler;
|
|
|
|
}
|
|
|
|
|
2008-12-09 14:13:29 +01:00
|
|
|
void CMakeTarget::clear()
|
|
|
|
{
|
|
|
|
executable = QString::null;
|
|
|
|
makeCommand = QString::null;
|
|
|
|
makeCleanCommand = QString::null;
|
|
|
|
workingDirectory = QString::null;
|
|
|
|
title = QString::null;
|
|
|
|
}
|
|
|
|
|