2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2013-01-28 17:12:19 +01:00
|
|
|
** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies).
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02: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.
|
|
|
|
|
**
|
|
|
|
|
** 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.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2008-12-02 16:19:05 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "projectfilewizardextension.h"
|
|
|
|
|
#include "projectexplorer.h"
|
2009-09-25 11:35:44 +02:00
|
|
|
#include "session.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "projectnodes.h"
|
|
|
|
|
#include "nodesvisitor.h"
|
|
|
|
|
#include "projectwizardpage.h"
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
#include <utils/stringutils.h>
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <coreplugin/basefilewizard.h>
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/iversioncontrol.h>
|
|
|
|
|
#include <coreplugin/vcsmanager.h>
|
2011-08-05 09:59:28 +02:00
|
|
|
#include <coreplugin/mimedatabase.h>
|
2010-01-12 16:45:21 +01:00
|
|
|
#include <extensionsystem/pluginmanager.h>
|
2011-08-05 09:59:28 +02:00
|
|
|
#include <texteditor/texteditorsettings.h>
|
2011-08-16 10:45:23 +02:00
|
|
|
#include <texteditor/icodestylepreferences.h>
|
2011-08-05 09:59:28 +02:00
|
|
|
#include <texteditor/icodestylepreferencesfactory.h>
|
|
|
|
|
#include <texteditor/normalindenter.h>
|
2011-08-16 10:45:23 +02:00
|
|
|
#include <texteditor/tabsettings.h>
|
2013-08-07 10:19:24 +02:00
|
|
|
#include <texteditor/storagesettings.h>
|
2011-08-05 09:59:28 +02:00
|
|
|
#include <projectexplorer/project.h>
|
|
|
|
|
#include <projectexplorer/editorconfiguration.h>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QtAlgorithms>
|
2013-06-05 14:04:07 +02:00
|
|
|
#include <QPointer>
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
|
|
|
|
#include <QFileInfo>
|
|
|
|
|
#include <QMultiMap>
|
|
|
|
|
#include <QDir>
|
|
|
|
|
#include <QTextDocument>
|
|
|
|
|
#include <QTextCursor>
|
2013-03-17 16:09:01 +01:00
|
|
|
#include <QMessageBox>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2011-04-14 12:58:14 +02:00
|
|
|
/*!
|
|
|
|
|
\class ProjectExplorer::Internal::ProjectFileWizardExtension
|
|
|
|
|
|
2013-06-05 14:29:24 +02:00
|
|
|
\brief The ProjectFileWizardExtension class implements the post-file
|
|
|
|
|
generating steps of a project wizard.
|
2011-04-14 12:58:14 +02:00
|
|
|
|
2013-09-10 17:16:10 +02:00
|
|
|
This class provides the following functions:
|
2011-04-14 12:58:14 +02:00
|
|
|
\list
|
2013-02-06 08:50:23 +01:00
|
|
|
\li Add to a project file (*.pri/ *.pro)
|
2013-09-10 17:16:10 +02:00
|
|
|
\li Initialize a version control system repository (unless the path is already
|
2011-04-14 12:58:14 +02:00
|
|
|
managed) and do 'add' if the VCS supports it.
|
|
|
|
|
\endlist
|
|
|
|
|
|
|
|
|
|
\sa ProjectExplorer::Internal::ProjectWizardPage
|
|
|
|
|
*/
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
enum { debugExtension = 0 };
|
|
|
|
|
|
|
|
|
|
namespace ProjectExplorer {
|
|
|
|
|
|
|
|
|
|
typedef QList<ProjectNode *> ProjectNodeList;
|
|
|
|
|
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2010-05-11 14:13:38 +02:00
|
|
|
// AllProjectNodesVisitor: Retrieve all projects (*.pri/*.pro)
|
|
|
|
|
// which support adding files
|
2008-12-02 12:01:29 +01:00
|
|
|
class AllProjectNodesVisitor : public NodesVisitor
|
|
|
|
|
{
|
|
|
|
|
public:
|
2010-08-24 17:17:11 +02:00
|
|
|
AllProjectNodesVisitor(ProjectNode::ProjectAction action)
|
|
|
|
|
: m_action(action)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
static ProjectNodeList allProjects(ProjectNode::ProjectAction action);
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
virtual void visitProjectNode(ProjectNode *node);
|
|
|
|
|
|
|
|
|
|
private:
|
2010-01-12 16:45:21 +01:00
|
|
|
ProjectNodeList m_projectNodes;
|
2010-08-24 17:17:11 +02:00
|
|
|
ProjectNode::ProjectAction m_action;
|
2008-12-02 12:01:29 +01:00
|
|
|
};
|
|
|
|
|
|
2010-08-24 17:17:11 +02:00
|
|
|
ProjectNodeList AllProjectNodesVisitor::allProjects(ProjectNode::ProjectAction action)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-08-24 17:17:11 +02:00
|
|
|
AllProjectNodesVisitor visitor(action);
|
2013-09-05 11:46:07 +02:00
|
|
|
SessionManager::sessionNode()->accept(&visitor);
|
2010-01-12 16:45:21 +01:00
|
|
|
return visitor.m_projectNodes;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void AllProjectNodesVisitor::visitProjectNode(ProjectNode *node)
|
|
|
|
|
{
|
2010-08-24 17:17:11 +02:00
|
|
|
if (node->supportedActions(node).contains(m_action))
|
2010-01-12 16:45:21 +01:00
|
|
|
m_projectNodes.push_back(node);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ProjectEntry: Context entry for a *.pri/*.pro file. Stores name and path
|
|
|
|
|
// for quick sort and path search, provides operator<() for maps.
|
|
|
|
|
struct ProjectEntry {
|
|
|
|
|
enum Type { ProFile, PriFile }; // Sort order: 'pro' before 'pri'
|
|
|
|
|
|
|
|
|
|
ProjectEntry() : node(0), type(ProFile) {}
|
|
|
|
|
explicit ProjectEntry(ProjectNode *node);
|
|
|
|
|
|
|
|
|
|
int compare(const ProjectEntry &rhs) const;
|
|
|
|
|
|
|
|
|
|
ProjectNode *node;
|
2010-09-30 16:50:59 +02:00
|
|
|
QString directory; // For matching against wizards' files, which are native.
|
2010-01-12 16:45:21 +01:00
|
|
|
QString fileName;
|
|
|
|
|
QString baseName;
|
|
|
|
|
Type type;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
ProjectEntry::ProjectEntry(ProjectNode *n) :
|
|
|
|
|
node(n),
|
|
|
|
|
type(ProFile)
|
|
|
|
|
{
|
|
|
|
|
const QFileInfo fi(node->path());
|
|
|
|
|
fileName = fi.fileName();
|
|
|
|
|
baseName = fi.baseName();
|
|
|
|
|
if (fi.suffix() != QLatin1String("pro"))
|
2010-01-29 22:49:55 +01:00
|
|
|
type = PriFile;
|
2010-09-30 16:50:59 +02:00
|
|
|
directory = fi.absolutePath();
|
2010-01-12 16:45:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Sort helper that sorts by base name and puts '*.pro' before '*.pri'
|
|
|
|
|
int ProjectEntry::compare(const ProjectEntry &rhs) const
|
|
|
|
|
{
|
2010-09-30 16:50:59 +02:00
|
|
|
if (const int drc = directory.compare(rhs.directory))
|
2010-03-15 15:37:08 +01:00
|
|
|
return drc;
|
2010-01-12 16:45:21 +01:00
|
|
|
if (const int brc = baseName.compare(rhs.baseName))
|
|
|
|
|
return brc;
|
|
|
|
|
if (type < rhs.type)
|
|
|
|
|
return -1;
|
|
|
|
|
if (type > rhs.type)
|
|
|
|
|
return 1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline bool operator<(const ProjectEntry &pe1, const ProjectEntry &pe2)
|
|
|
|
|
{
|
|
|
|
|
return pe1.compare(pe2) < 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QDebug operator<<(QDebug d, const ProjectEntry &e)
|
|
|
|
|
{
|
2010-09-30 16:50:59 +02:00
|
|
|
d.nospace() << e.directory << ' ' << e.fileName << ' ' << e.type;
|
2010-01-12 16:45:21 +01:00
|
|
|
return d;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// --------- ProjectWizardContext
|
2009-01-20 15:31:33 +01:00
|
|
|
struct ProjectWizardContext
|
|
|
|
|
{
|
2010-01-12 16:45:21 +01:00
|
|
|
ProjectWizardContext();
|
|
|
|
|
void clear();
|
|
|
|
|
|
|
|
|
|
QList<Core::IVersionControl*> versionControls;
|
2011-04-15 17:43:44 +02:00
|
|
|
QList<Core::IVersionControl*> activeVersionControls;
|
2010-01-12 16:45:21 +01:00
|
|
|
QList<ProjectEntry> projects;
|
2013-06-05 13:41:47 +02:00
|
|
|
QPointer<ProjectWizardPage> page; // this is managed by the wizard!
|
2010-01-12 16:45:21 +01:00
|
|
|
bool repositoryExists; // Is VCS 'add' sufficient, or should a repository be created?
|
|
|
|
|
QString commonDirectory;
|
2010-08-26 18:33:16 +02:00
|
|
|
const Core::IWizard *wizard;
|
2008-12-02 12:01:29 +01:00
|
|
|
};
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
ProjectWizardContext::ProjectWizardContext() :
|
|
|
|
|
page(0),
|
2010-08-26 18:33:16 +02:00
|
|
|
repositoryExists(false),
|
|
|
|
|
wizard(0)
|
2010-01-12 16:45:21 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProjectWizardContext::clear()
|
|
|
|
|
{
|
2011-04-15 17:43:44 +02:00
|
|
|
activeVersionControls.clear();
|
2010-01-12 16:45:21 +01:00
|
|
|
projects.clear();
|
|
|
|
|
commonDirectory.clear();
|
2013-06-05 14:04:07 +02:00
|
|
|
page = 0;
|
2010-01-12 16:45:21 +01:00
|
|
|
repositoryExists = false;
|
2010-08-26 18:33:16 +02:00
|
|
|
wizard = 0;
|
2010-01-12 16:45:21 +01:00
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
// ---- ProjectFileWizardExtension
|
2009-01-20 15:31:33 +01:00
|
|
|
ProjectFileWizardExtension::ProjectFileWizardExtension()
|
|
|
|
|
: m_context(0)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ProjectFileWizardExtension::~ProjectFileWizardExtension()
|
|
|
|
|
{
|
|
|
|
|
delete m_context;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-24 17:17:11 +02:00
|
|
|
static QList<ProjectEntry> findDeployProject(const QList<ProjectEntry> &projects,
|
|
|
|
|
QString &commonPath)
|
|
|
|
|
{
|
|
|
|
|
QList<ProjectEntry> filtered;
|
|
|
|
|
foreach (const ProjectEntry &project, projects)
|
|
|
|
|
if (project.node->deploysFolder(commonPath))
|
|
|
|
|
filtered << project;
|
|
|
|
|
return filtered;
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
// Find the project the new files should be added to given their common
|
|
|
|
|
// path. Either a direct match on the directory or the directory with
|
|
|
|
|
// the longest matching path (list containing"/project/subproject1" matching
|
|
|
|
|
// common path "/project/subproject1/newuserpath").
|
|
|
|
|
static int findMatchingProject(const QList<ProjectEntry> &projects,
|
2012-04-02 14:55:56 +02:00
|
|
|
const QString &commonPath,
|
|
|
|
|
const QString &preferedProjectNode)
|
2010-01-12 16:45:21 +01:00
|
|
|
{
|
|
|
|
|
if (projects.isEmpty() || commonPath.isEmpty())
|
|
|
|
|
return -1;
|
|
|
|
|
|
2012-04-02 14:55:56 +02:00
|
|
|
const int count = projects.size();
|
|
|
|
|
if (!preferedProjectNode.isEmpty()) {
|
|
|
|
|
for (int p = 0; p < count; ++p) {
|
|
|
|
|
if (projects.at(p).node->path() == preferedProjectNode)
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
int bestMatch = -1;
|
|
|
|
|
int bestMatchLength = 0;
|
2010-10-19 13:59:02 +02:00
|
|
|
bool bestMatchIsProFile = false;
|
2010-01-12 16:45:21 +01:00
|
|
|
for (int p = 0; p < count; p++) {
|
|
|
|
|
// Direct match or better match? (note that the wizards' files are native).
|
2010-10-19 13:59:02 +02:00
|
|
|
const ProjectEntry &entry = projects.at(p);
|
|
|
|
|
const QString &projectDirectory = entry.directory;
|
|
|
|
|
const int projectDirectorySize = projectDirectory.size();
|
|
|
|
|
if (projectDirectorySize == bestMatchLength && bestMatchIsProFile)
|
|
|
|
|
continue; // prefer first pro file over all other files with same bestMatchLength
|
|
|
|
|
if (projectDirectorySize == bestMatchLength && entry.type == ProjectEntry::PriFile)
|
|
|
|
|
continue; // we already have a match with same bestMatchLength that is at least a pri file
|
|
|
|
|
if (projectDirectorySize >= bestMatchLength
|
|
|
|
|
&& commonPath.startsWith(projectDirectory)) {
|
|
|
|
|
bestMatchIsProFile = (entry.type == ProjectEntry::ProFile);
|
2010-01-12 16:45:21 +01:00
|
|
|
bestMatchLength = projectDirectory.size();
|
|
|
|
|
bestMatch = p;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return bestMatch;
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-02 12:14:27 +02:00
|
|
|
static QString generatedProjectFilePath(const QList<Core::GeneratedFile> &files)
|
|
|
|
|
{
|
2011-05-10 15:19:38 +02:00
|
|
|
foreach (const Core::GeneratedFile &file, files)
|
2010-09-02 12:14:27 +02:00
|
|
|
if (file.attributes() & Core::GeneratedFile::OpenProjectAttribute)
|
|
|
|
|
return file.path();
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-26 18:33:16 +02:00
|
|
|
void ProjectFileWizardExtension::firstExtensionPageShown(
|
2012-04-02 14:55:56 +02:00
|
|
|
const QList<Core::GeneratedFile> &files,
|
|
|
|
|
const QVariantMap &extraValues)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-09-02 12:14:27 +02:00
|
|
|
initProjectChoices(generatedProjectFilePath(files));
|
2010-08-26 18:33:16 +02:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
if (debugExtension)
|
|
|
|
|
qDebug() << Q_FUNC_INFO << files.size();
|
2010-01-12 16:45:21 +01:00
|
|
|
// Parametrize wizard page: find best project to add to, set up files display and
|
|
|
|
|
// version control depending on path
|
2008-12-02 12:01:29 +01:00
|
|
|
QStringList fileNames;
|
|
|
|
|
foreach (const Core::GeneratedFile &f, files)
|
|
|
|
|
fileNames.push_back(f.path());
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->commonDirectory = Utils::commonPath(fileNames);
|
|
|
|
|
m_context->page->setFilesDisplay(m_context->commonDirectory, fileNames);
|
|
|
|
|
// Find best project (Entry at 0 is 'None').
|
2010-08-24 17:17:11 +02:00
|
|
|
|
|
|
|
|
int bestProjectIndex = -1;
|
|
|
|
|
|
|
|
|
|
QList<ProjectEntry> deployingProjects = findDeployProject(m_context->projects, m_context->commonDirectory);
|
|
|
|
|
if (!deployingProjects.isEmpty()) {
|
|
|
|
|
// Oh we do have someone that deploys it
|
|
|
|
|
// then the best match is NONE
|
|
|
|
|
// We display a label explaining that and rename <None> to
|
2010-09-10 10:51:43 +02:00
|
|
|
// <Implicitly Add>
|
|
|
|
|
m_context->page->setNoneLabel(tr("<Implicitly Add>"));
|
2010-08-24 17:17:11 +02:00
|
|
|
|
|
|
|
|
QString text = tr("The files are implicitly added to the projects:\n");
|
2010-09-10 10:51:43 +02:00
|
|
|
foreach (const ProjectEntry &project, deployingProjects) {
|
|
|
|
|
text += project.fileName;
|
|
|
|
|
text += QLatin1Char('\n');
|
|
|
|
|
}
|
2010-08-24 17:17:11 +02:00
|
|
|
|
|
|
|
|
m_context->page->setAdditionalInfo(text);
|
|
|
|
|
bestProjectIndex = -1;
|
|
|
|
|
} else {
|
2012-04-02 14:55:56 +02:00
|
|
|
bestProjectIndex = findMatchingProject(m_context->projects, m_context->commonDirectory,
|
|
|
|
|
extraValues.value(QLatin1String(Constants::PREFERED_PROJECT_NODE)).toString());
|
2010-08-24 17:17:11 +02:00
|
|
|
m_context->page->setNoneLabel(tr("<None>"));
|
|
|
|
|
}
|
|
|
|
|
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
if (bestProjectIndex == -1)
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->page->setCurrentProjectIndex(0);
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
else
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->page->setCurrentProjectIndex(bestProjectIndex + 1);
|
2011-04-15 17:43:44 +02:00
|
|
|
|
|
|
|
|
// Store all version controls for later use:
|
2011-07-27 12:36:29 +00:00
|
|
|
if (m_context->versionControls.isEmpty()) {
|
2012-06-18 11:34:15 +02:00
|
|
|
foreach (Core::IVersionControl *vc, ExtensionSystem::PluginManager::getObjects<Core::IVersionControl>()) {
|
2011-07-27 12:36:29 +00:00
|
|
|
m_context->versionControls.append(vc);
|
|
|
|
|
connect(vc, SIGNAL(configurationChanged()), this, SLOT(initializeVersionControlChoices()));
|
|
|
|
|
}
|
2011-04-15 17:43:44 +02:00
|
|
|
}
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
initializeVersionControlChoices();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
void ProjectFileWizardExtension::initializeVersionControlChoices()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2013-06-05 13:41:47 +02:00
|
|
|
if (m_context->page.isNull())
|
|
|
|
|
return;
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
// Figure out version control situation:
|
|
|
|
|
// 1) Directory is managed and VCS supports "Add" -> List it
|
|
|
|
|
// 2) Directory is managed and VCS does not support "Add" -> None available
|
|
|
|
|
// 3) Directory is not managed -> Offer all VCS that support "CreateRepository"
|
2011-04-15 17:43:44 +02:00
|
|
|
|
|
|
|
|
Core::IVersionControl *currentSelection = 0;
|
|
|
|
|
int currentIdx = m_context->page->versionControlIndex() - 1;
|
|
|
|
|
if (currentIdx >= 0 && currentIdx <= m_context->activeVersionControls.size() - 1)
|
|
|
|
|
currentSelection = m_context->activeVersionControls.at(currentIdx);
|
|
|
|
|
|
|
|
|
|
m_context->activeVersionControls.clear();
|
|
|
|
|
|
|
|
|
|
QStringList versionControlChoices = QStringList(tr("<None>"));
|
2010-01-12 16:45:21 +01:00
|
|
|
if (!m_context->commonDirectory.isEmpty()) {
|
2013-08-30 17:13:29 +02:00
|
|
|
Core::IVersionControl *managingControl = Core::VcsManager::findVersionControlForDirectory(m_context->commonDirectory);
|
2010-01-12 16:45:21 +01:00
|
|
|
if (managingControl) {
|
|
|
|
|
// Under VCS
|
|
|
|
|
if (managingControl->supportsOperation(Core::IVersionControl::AddOperation)) {
|
2011-04-18 16:24:29 +02:00
|
|
|
versionControlChoices.append(managingControl->displayName());
|
2011-04-15 17:43:44 +02:00
|
|
|
m_context->activeVersionControls.push_back(managingControl);
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->repositoryExists = true;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Create
|
2011-04-15 17:43:44 +02:00
|
|
|
foreach (Core::IVersionControl *vc, m_context->versionControls)
|
|
|
|
|
if (vc->supportsOperation(Core::IVersionControl::CreateRepositoryOperation)) {
|
|
|
|
|
versionControlChoices.append(vc->displayName());
|
|
|
|
|
m_context->activeVersionControls.append(vc);
|
|
|
|
|
}
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->repositoryExists = false;
|
|
|
|
|
}
|
|
|
|
|
} // has a common root.
|
2011-04-15 17:43:44 +02:00
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->page->setVersionControls(versionControlChoices);
|
|
|
|
|
// Enable adding to version control by default.
|
|
|
|
|
if (m_context->repositoryExists && versionControlChoices.size() >= 2)
|
|
|
|
|
m_context->page->setVersionControlIndex(1);
|
2011-04-15 17:43:44 +02:00
|
|
|
if (!m_context->repositoryExists) {
|
|
|
|
|
int newIdx = m_context->activeVersionControls.indexOf(currentSelection) + 1;
|
|
|
|
|
m_context->page->setVersionControlIndex(newIdx);
|
|
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<QWizardPage *> ProjectFileWizardExtension::extensionPages(const Core::IWizard *wizard)
|
|
|
|
|
{
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
if (!m_context)
|
2008-12-02 12:01:29 +01:00
|
|
|
m_context = new ProjectWizardContext;
|
Remove braces for single lines of conditions
#!/usr/bin/env ruby
Dir.glob('**/*.cpp') { |file|
# skip ast (excluding paste, astpath, and canv'ast'imer)
next if file =~ /ast[^eip]|keywords\.|qualifiers|preprocessor|names.cpp/i
s = File.read(file)
next if s.include?('qlalr')
orig = s.dup
s.gsub!(/\n *if [^\n]*{\n[^\n]*\n\s+}(\s+else if [^\n]* {\n[^\n]*\n\s+})*(\s+else {\n[^\n]*\n\s+})?\n/m) { |m|
res = $&
if res =~ /^\s*(\/\/|[A-Z_]{3,})/ # C++ comment or macro (Q_UNUSED, SDEBUG), do not touch braces
res
else
res.gsub!('} else', 'else')
res.gsub!(/\n +} *\n/m, "\n")
res.gsub(/ *{$/, '')
end
}
s.gsub!(/ *$/, '')
File.open(file, 'wb').write(s) if s != orig
}
Change-Id: I3b30ee60df0986f66c02132c65fc38a3fbb6bbdc
Reviewed-by: hjk <qthjk@ovi.com>
2013-01-08 03:32:53 +02:00
|
|
|
else
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->clear();
|
2008-12-02 12:01:29 +01:00
|
|
|
// Init context with page and projects
|
|
|
|
|
m_context->page = new ProjectWizardPage;
|
2010-08-26 18:33:16 +02:00
|
|
|
m_context->wizard = wizard;
|
2010-01-12 16:45:21 +01:00
|
|
|
return QList<QWizardPage *>() << m_context->page;
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-13 16:37:48 +02:00
|
|
|
|
|
|
|
|
static inline void getProjectChoicesAndToolTips(QStringList *projectChoicesParam,
|
|
|
|
|
QStringList *projectToolTipsParam,
|
|
|
|
|
ProjectNode::ProjectAction *projectActionParam,
|
|
|
|
|
const QString &generatedProjectFilePath,
|
|
|
|
|
ProjectWizardContext *context)
|
2010-01-12 16:45:21 +01:00
|
|
|
{
|
2008-12-02 12:01:29 +01:00
|
|
|
// Set up project list which remains the same over duration of wizard execution
|
2010-03-15 15:37:08 +01:00
|
|
|
// As tooltip, set the directory for disambiguation (should someone have
|
|
|
|
|
// duplicate base names in differing directories).
|
2010-01-12 16:45:21 +01:00
|
|
|
//: No project selected
|
2012-04-13 16:37:48 +02:00
|
|
|
QStringList projectChoices(ProjectFileWizardExtension::tr("<None>"));
|
2010-06-14 10:35:22 +02:00
|
|
|
QStringList projectToolTips((QString()));
|
2010-08-24 17:17:11 +02:00
|
|
|
|
2010-08-26 18:33:16 +02:00
|
|
|
typedef QMap<ProjectEntry, bool> ProjectEntryMap;
|
|
|
|
|
// Sort by base name and purge duplicated entries (resulting from dependencies)
|
|
|
|
|
// via Map.
|
|
|
|
|
ProjectEntryMap entryMap;
|
|
|
|
|
|
|
|
|
|
ProjectNode::ProjectAction projectAction =
|
2012-04-13 16:37:48 +02:00
|
|
|
context->wizard->kind() == Core::IWizard::ProjectWizard
|
2010-08-26 18:33:16 +02:00
|
|
|
? ProjectNode::AddSubProject : ProjectNode::AddNewFile;
|
2012-11-28 20:44:03 +02:00
|
|
|
foreach (ProjectNode *n, AllProjectNodesVisitor::allProjects(projectAction)) {
|
2010-08-26 18:33:16 +02:00
|
|
|
if (projectAction == ProjectNode::AddNewFile
|
|
|
|
|
|| (projectAction == ProjectNode::AddSubProject
|
2012-04-13 16:37:48 +02:00
|
|
|
&& (generatedProjectFilePath.isEmpty() ? true : n->canAddSubProject(generatedProjectFilePath))))
|
2010-01-12 16:45:21 +01:00
|
|
|
entryMap.insert(ProjectEntry(n), true);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2010-08-26 18:33:16 +02:00
|
|
|
|
2012-04-13 16:37:48 +02:00
|
|
|
context->projects.clear();
|
2010-08-26 18:33:16 +02:00
|
|
|
|
|
|
|
|
// Collect names
|
|
|
|
|
const ProjectEntryMap::const_iterator cend = entryMap.constEnd();
|
|
|
|
|
for (ProjectEntryMap::const_iterator it = entryMap.constBegin(); it != cend; ++it) {
|
2012-04-13 16:37:48 +02:00
|
|
|
context->projects.push_back(it.key());
|
2010-08-26 18:33:16 +02:00
|
|
|
projectChoices.push_back(it.key().fileName);
|
2010-09-30 16:50:59 +02:00
|
|
|
projectToolTips.push_back(QDir::toNativeSeparators(it.key().directory));
|
2010-08-26 18:33:16 +02:00
|
|
|
}
|
2012-04-13 16:37:48 +02:00
|
|
|
*projectChoicesParam = projectChoices;
|
|
|
|
|
*projectToolTipsParam = projectToolTips;
|
|
|
|
|
*projectActionParam = projectAction;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProjectFileWizardExtension::initProjectChoices(const QString &generatedProjectFilePath)
|
|
|
|
|
{
|
|
|
|
|
QStringList projectChoices;
|
|
|
|
|
QStringList projectToolTips;
|
|
|
|
|
ProjectNode::ProjectAction projectAction;
|
|
|
|
|
|
|
|
|
|
getProjectChoicesAndToolTips(&projectChoices, &projectToolTips, &projectAction,
|
|
|
|
|
generatedProjectFilePath, m_context);
|
2010-08-26 18:33:16 +02:00
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
m_context->page->setProjects(projectChoices);
|
2010-03-15 15:37:08 +01:00
|
|
|
m_context->page->setProjectToolTips(projectToolTips);
|
2011-08-29 15:31:33 +02:00
|
|
|
m_context->page->setAddingSubProject(projectAction == ProjectNode::AddSubProject);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2011-08-05 09:59:28 +02:00
|
|
|
bool ProjectFileWizardExtension::processFiles(
|
2010-08-26 18:33:16 +02:00
|
|
|
const QList<Core::GeneratedFile> &files,
|
|
|
|
|
bool *removeOpenProjectAttribute, QString *errorMessage)
|
2010-01-12 16:45:21 +01:00
|
|
|
{
|
2013-03-17 16:09:01 +01:00
|
|
|
if (!processProject(files, removeOpenProjectAttribute, errorMessage))
|
|
|
|
|
return false;
|
|
|
|
|
if (!processVersionControl(files, errorMessage)) {
|
|
|
|
|
QString message;
|
|
|
|
|
if (errorMessage) {
|
|
|
|
|
message = *errorMessage;
|
|
|
|
|
message.append(QLatin1String("\n\n"));
|
|
|
|
|
errorMessage->clear();
|
|
|
|
|
}
|
|
|
|
|
message.append(tr("Open project anyway?"));
|
|
|
|
|
if (QMessageBox::question(Core::ICore::mainWindow(), tr("Version Control Failure"), message,
|
|
|
|
|
QMessageBox::Yes, QMessageBox::No) == QMessageBox::No)
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2010-01-12 16:45:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add files to project && version control
|
2010-08-26 18:33:16 +02:00
|
|
|
bool ProjectFileWizardExtension::processProject(
|
|
|
|
|
const QList<Core::GeneratedFile> &files,
|
|
|
|
|
bool *removeOpenProjectAttribute, QString *errorMessage)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2010-08-26 18:33:16 +02:00
|
|
|
*removeOpenProjectAttribute = false;
|
|
|
|
|
|
2010-09-02 12:14:27 +02:00
|
|
|
QString generatedProject = generatedProjectFilePath(files);
|
|
|
|
|
|
2010-01-12 16:45:21 +01:00
|
|
|
// Add files to project (Entry at 0 is 'None').
|
|
|
|
|
const int projectIndex = m_context->page->currentProjectIndex() - 1;
|
|
|
|
|
if (projectIndex < 0 || projectIndex >= m_context->projects.size())
|
|
|
|
|
return true;
|
|
|
|
|
ProjectNode *project = m_context->projects.at(projectIndex).node;
|
2010-08-26 18:33:16 +02:00
|
|
|
if (m_context->wizard->kind() == Core::IWizard::ProjectWizard) {
|
2010-09-02 12:14:27 +02:00
|
|
|
if (!project->addSubProjects(QStringList(generatedProject))) {
|
2010-08-26 18:33:16 +02:00
|
|
|
*errorMessage = tr("Failed to add subproject '%1'\nto project '%2'.")
|
2010-09-02 12:14:27 +02:00
|
|
|
.arg(generatedProject).arg(project->path());
|
2010-01-12 16:45:21 +01:00
|
|
|
return false;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
2010-08-26 18:33:16 +02:00
|
|
|
*removeOpenProjectAttribute = true;
|
|
|
|
|
} else {
|
2013-07-01 16:13:48 +02:00
|
|
|
QStringList filePaths;
|
|
|
|
|
foreach (const Core::GeneratedFile &generatedFile, files)
|
|
|
|
|
filePaths << generatedFile.path();
|
|
|
|
|
if (!project->addFiles(filePaths)) {
|
|
|
|
|
*errorMessage = tr("Failed to add one or more files to project\n'%1' (%2).").
|
|
|
|
|
arg(project->path(), filePaths.join(QString(QLatin1Char(','))));
|
|
|
|
|
return false;
|
2010-08-26 18:33:16 +02:00
|
|
|
}
|
2010-01-12 16:45:21 +01:00
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ProjectFileWizardExtension::processVersionControl(const QList<Core::GeneratedFile> &files, QString *errorMessage)
|
|
|
|
|
{
|
|
|
|
|
// Add files to version control (Entry at 0 is 'None').
|
|
|
|
|
const int vcsIndex = m_context->page->versionControlIndex() - 1;
|
2011-04-15 17:43:44 +02:00
|
|
|
if (vcsIndex < 0 || vcsIndex >= m_context->activeVersionControls.size())
|
2010-01-12 16:45:21 +01:00
|
|
|
return true;
|
|
|
|
|
QTC_ASSERT(!m_context->commonDirectory.isEmpty(), return false);
|
2011-04-15 17:43:44 +02:00
|
|
|
Core::IVersionControl *versionControl = m_context->activeVersionControls.at(vcsIndex);
|
2010-01-12 16:45:21 +01:00
|
|
|
// Create repository?
|
|
|
|
|
if (!m_context->repositoryExists) {
|
|
|
|
|
QTC_ASSERT(versionControl->supportsOperation(Core::IVersionControl::CreateRepositoryOperation), return false);
|
|
|
|
|
if (!versionControl->vcsCreateRepository(m_context->commonDirectory)) {
|
|
|
|
|
*errorMessage = tr("A version control system repository could not be created in '%1'.").arg(m_context->commonDirectory);
|
|
|
|
|
return false;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
}
|
2010-01-12 16:45:21 +01:00
|
|
|
// Add files if supported.
|
|
|
|
|
if (versionControl->supportsOperation(Core::IVersionControl::AddOperation)) {
|
2008-12-02 12:01:29 +01:00
|
|
|
foreach (const Core::GeneratedFile &generatedFile, files) {
|
2010-01-12 16:45:21 +01:00
|
|
|
if (!versionControl->vcsAdd(generatedFile.path())) {
|
2008-12-02 12:01:29 +01:00
|
|
|
*errorMessage = tr("Failed to add '%1' to the version control system.").arg(generatedFile.path());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-21 21:47:17 +02:00
|
|
|
static TextEditor::ICodeStylePreferences *codeStylePreferences(ProjectExplorer::Project *project, Core::Id languageId)
|
2011-08-05 09:59:28 +02:00
|
|
|
{
|
2012-11-21 21:47:17 +02:00
|
|
|
if (!languageId.isValid())
|
2011-08-05 09:59:28 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (project)
|
2011-08-16 10:45:23 +02:00
|
|
|
return project->editorConfiguration()->codeStyle(languageId);
|
2011-08-05 09:59:28 +02:00
|
|
|
|
2013-09-19 17:59:27 +02:00
|
|
|
return TextEditor::TextEditorSettings::codeStyle(languageId);
|
2011-08-05 09:59:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProjectFileWizardExtension::applyCodeStyle(Core::GeneratedFile *file) const
|
|
|
|
|
{
|
|
|
|
|
if (file->isBinary() || file->contents().isEmpty())
|
|
|
|
|
return; // nothing to do
|
|
|
|
|
|
2013-08-30 16:38:57 +02:00
|
|
|
Core::MimeType mt = Core::MimeDatabase::findByFile(QFileInfo(file->path()));
|
2013-09-19 17:59:27 +02:00
|
|
|
Core::Id languageId = TextEditor::TextEditorSettings::languageId(mt.type());
|
2011-08-05 09:59:28 +02:00
|
|
|
|
2012-11-21 21:47:17 +02:00
|
|
|
if (!languageId.isValid())
|
2011-08-05 09:59:28 +02:00
|
|
|
return; // don't modify files like *.ui *.pro
|
|
|
|
|
|
|
|
|
|
ProjectNode *project = 0;
|
|
|
|
|
const int projectIndex = m_context->page->currentProjectIndex() - 1;
|
|
|
|
|
if (projectIndex >= 0 && projectIndex < m_context->projects.size())
|
|
|
|
|
project = m_context->projects.at(projectIndex).node;
|
|
|
|
|
|
2013-09-05 11:46:07 +02:00
|
|
|
Project *baseProject = SessionManager::projectForNode(project);
|
2011-08-05 09:59:28 +02:00
|
|
|
|
|
|
|
|
TextEditor::ICodeStylePreferencesFactory *factory
|
2013-09-19 17:59:27 +02:00
|
|
|
= TextEditor::TextEditorSettings::codeStyleFactory(languageId);
|
2011-08-05 09:59:28 +02:00
|
|
|
|
|
|
|
|
TextEditor::Indenter *indenter = 0;
|
|
|
|
|
if (factory)
|
|
|
|
|
indenter = factory->createIndenter();
|
|
|
|
|
if (!indenter)
|
|
|
|
|
indenter = new TextEditor::NormalIndenter();
|
|
|
|
|
|
2011-08-16 10:45:23 +02:00
|
|
|
TextEditor::ICodeStylePreferences *codeStylePrefs = codeStylePreferences(baseProject, languageId);
|
2011-08-05 09:59:28 +02:00
|
|
|
indenter->setCodeStylePreferences(codeStylePrefs);
|
|
|
|
|
QTextDocument doc(file->contents());
|
|
|
|
|
QTextCursor cursor(&doc);
|
|
|
|
|
cursor.select(QTextCursor::Document);
|
2011-08-16 10:45:23 +02:00
|
|
|
indenter->indent(&doc, cursor, QChar::Null, codeStylePrefs->currentTabSettings());
|
2011-08-05 09:59:28 +02:00
|
|
|
delete indenter;
|
2013-09-19 17:59:27 +02:00
|
|
|
if (TextEditor::TextEditorSettings::storageSettings().m_cleanWhitespace) {
|
2013-08-07 10:19:24 +02:00
|
|
|
QTextBlock block = doc.firstBlock();
|
|
|
|
|
while (block.isValid()) {
|
|
|
|
|
codeStylePrefs->currentTabSettings().removeTrailingWhitespace(cursor, block);
|
|
|
|
|
block = block.next();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
file->setContents(doc.toPlainText());
|
2011-08-05 09:59:28 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 16:37:48 +02:00
|
|
|
QStringList ProjectFileWizardExtension::getProjectChoices() const
|
|
|
|
|
{
|
|
|
|
|
QStringList projectChoices;
|
|
|
|
|
QStringList projectToolTips;
|
|
|
|
|
ProjectNode::ProjectAction projectAction;
|
|
|
|
|
|
|
|
|
|
getProjectChoicesAndToolTips(&projectChoices, &projectToolTips, &projectAction,
|
|
|
|
|
QString(), m_context);
|
|
|
|
|
|
|
|
|
|
return projectChoices;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QStringList ProjectFileWizardExtension::getProjectToolTips() const
|
|
|
|
|
{
|
|
|
|
|
QStringList projectChoices;
|
|
|
|
|
QStringList projectToolTips;
|
|
|
|
|
ProjectNode::ProjectAction projectAction;
|
|
|
|
|
|
|
|
|
|
getProjectChoicesAndToolTips(&projectChoices, &projectToolTips, &projectAction,
|
|
|
|
|
QString(), m_context);
|
|
|
|
|
|
|
|
|
|
return projectToolTips;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProjectFileWizardExtension::hideProjectComboBox()
|
|
|
|
|
{
|
|
|
|
|
m_context->page->setProjectComoBoxVisible(false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ProjectFileWizardExtension::setProjectIndex(int i)
|
|
|
|
|
{
|
|
|
|
|
m_context->page->setCurrentProjectIndex(i);
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 16:19:05 +01:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace ProjectExplorer
|