2013-01-14 23:11:10 +04:00
|
|
|
/****************************************************************************
|
|
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2013-01-14 23:11:10 +04:00
|
|
|
**
|
|
|
|
|
** 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
|
2016-01-15 14:57:40 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2013-01-14 23:11:10 +04:00
|
|
|
**
|
2016-01-15 14:57:40 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2013-01-14 23:11:10 +04:00
|
|
|
**
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "pythoneditorplugin.h"
|
2014-08-20 15:05:32 +02:00
|
|
|
#include "pythoneditor.h"
|
2013-01-14 23:11:10 +04:00
|
|
|
#include "pythoneditorconstants.h"
|
2014-06-26 00:27:27 +02:00
|
|
|
#include "tools/pythonhighlighter.h"
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
#include <coreplugin/coreconstants.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
#include <coreplugin/documentmanager.h>
|
2013-01-14 23:11:10 +04:00
|
|
|
#include <coreplugin/fileiconprovider.h>
|
|
|
|
|
#include <coreplugin/id.h>
|
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2013-01-14 23:11:10 +04:00
|
|
|
#include <extensionsystem/pluginmanager.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
#include <projectexplorer/applicationlauncher.h>
|
|
|
|
|
#include <projectexplorer/kitmanager.h>
|
|
|
|
|
#include <projectexplorer/localenvironmentaspect.h>
|
|
|
|
|
#include <projectexplorer/runconfiguration.h>
|
|
|
|
|
#include <projectexplorer/runconfigurationaspects.h>
|
|
|
|
|
#include <projectexplorer/project.h>
|
2016-01-27 16:56:36 +01:00
|
|
|
#include <projectexplorer/runnables.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
#include <projectexplorer/target.h>
|
|
|
|
|
#include <projectexplorer/iprojectmanager.h>
|
|
|
|
|
#include <projectexplorer/projectnodes.h>
|
2015-11-23 16:41:54 +01:00
|
|
|
#include <projectexplorer/projectexplorericons.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2013-01-14 23:11:10 +04:00
|
|
|
#include <texteditor/texteditorconstants.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
#include <utils/detailswidget.h>
|
2015-02-04 09:32:46 +01:00
|
|
|
#include <utils/mimetypes/mimedatabase.h>
|
2015-06-09 15:19:54 +02:00
|
|
|
#include <utils/pathchooser.h>
|
|
|
|
|
#include <utils/qtcprocess.h>
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
#include <QtPlugin>
|
|
|
|
|
#include <QCoreApplication>
|
2015-06-09 15:19:54 +02:00
|
|
|
#include <QFormLayout>
|
2013-01-14 23:11:10 +04:00
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
using namespace Core;
|
|
|
|
|
using namespace ProjectExplorer;
|
2013-01-14 23:11:10 +04:00
|
|
|
using namespace PythonEditor::Constants;
|
2015-06-09 15:19:54 +02:00
|
|
|
using namespace Utils;
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
|
* List of Python keywords (includes "print" that isn't keyword in python 3
|
|
|
|
|
******************************************************************************/
|
|
|
|
|
static const char *const LIST_OF_PYTHON_KEYWORDS[] = {
|
|
|
|
|
"and",
|
|
|
|
|
"as",
|
|
|
|
|
"assert",
|
|
|
|
|
"break",
|
|
|
|
|
"class",
|
|
|
|
|
"continue",
|
|
|
|
|
"def",
|
|
|
|
|
"del",
|
|
|
|
|
"elif",
|
|
|
|
|
"else",
|
|
|
|
|
"except",
|
|
|
|
|
"exec",
|
|
|
|
|
"finally",
|
|
|
|
|
"for",
|
|
|
|
|
"from",
|
|
|
|
|
"global",
|
|
|
|
|
"if",
|
|
|
|
|
"import",
|
|
|
|
|
"in",
|
|
|
|
|
"is",
|
|
|
|
|
"lambda",
|
|
|
|
|
"not",
|
|
|
|
|
"or",
|
|
|
|
|
"pass",
|
|
|
|
|
"print",
|
|
|
|
|
"raise",
|
|
|
|
|
"return",
|
|
|
|
|
"try",
|
|
|
|
|
"while",
|
|
|
|
|
"with",
|
|
|
|
|
"yield"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
|
* List of Python magic methods and attributes
|
|
|
|
|
******************************************************************************/
|
|
|
|
|
static const char *const LIST_OF_PYTHON_MAGICS[] = {
|
|
|
|
|
// ctor & dtor
|
|
|
|
|
"__init__",
|
|
|
|
|
"__del__",
|
|
|
|
|
// string conversion functions
|
|
|
|
|
"__str__",
|
|
|
|
|
"__repr__",
|
|
|
|
|
"__unicode__",
|
|
|
|
|
// attribute access functions
|
|
|
|
|
"__setattr__",
|
|
|
|
|
"__getattr__",
|
|
|
|
|
"__delattr__",
|
|
|
|
|
// binary operators
|
|
|
|
|
"__add__",
|
|
|
|
|
"__sub__",
|
|
|
|
|
"__mul__",
|
|
|
|
|
"__truediv__",
|
|
|
|
|
"__floordiv__",
|
|
|
|
|
"__mod__",
|
|
|
|
|
"__pow__",
|
|
|
|
|
"__and__",
|
|
|
|
|
"__or__",
|
|
|
|
|
"__xor__",
|
|
|
|
|
"__eq__",
|
|
|
|
|
"__ne__",
|
|
|
|
|
"__gt__",
|
|
|
|
|
"__lt__",
|
|
|
|
|
"__ge__",
|
|
|
|
|
"__le__",
|
|
|
|
|
"__lshift__",
|
|
|
|
|
"__rshift__",
|
|
|
|
|
"__contains__",
|
|
|
|
|
// unary operators
|
|
|
|
|
"__pos__",
|
|
|
|
|
"__neg__",
|
|
|
|
|
"__inv__",
|
|
|
|
|
"__abs__",
|
|
|
|
|
"__len__",
|
|
|
|
|
// item operators like []
|
|
|
|
|
"__getitem__",
|
|
|
|
|
"__setitem__",
|
|
|
|
|
"__delitem__",
|
|
|
|
|
"__getslice__",
|
|
|
|
|
"__setslice__",
|
|
|
|
|
"__delslice__",
|
|
|
|
|
// other functions
|
|
|
|
|
"__cmp__",
|
|
|
|
|
"__hash__",
|
|
|
|
|
"__nonzero__",
|
|
|
|
|
"__call__",
|
|
|
|
|
"__iter__",
|
|
|
|
|
"__reversed__",
|
|
|
|
|
"__divmod__",
|
|
|
|
|
"__int__",
|
|
|
|
|
"__long__",
|
|
|
|
|
"__float__",
|
|
|
|
|
"__complex__",
|
|
|
|
|
"__hex__",
|
|
|
|
|
"__oct__",
|
|
|
|
|
"__index__",
|
|
|
|
|
"__copy__",
|
|
|
|
|
"__deepcopy__",
|
|
|
|
|
"__sizeof__",
|
|
|
|
|
"__trunc__",
|
|
|
|
|
"__format__",
|
|
|
|
|
// magic attributes
|
|
|
|
|
"__name__",
|
|
|
|
|
"__module__",
|
|
|
|
|
"__dict__",
|
|
|
|
|
"__bases__",
|
|
|
|
|
"__doc__"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
|
* List of python built-in functions and objects
|
|
|
|
|
******************************************************************************/
|
|
|
|
|
static const char *const LIST_OF_PYTHON_BUILTINS[] = {
|
|
|
|
|
"range",
|
|
|
|
|
"xrange",
|
|
|
|
|
"int",
|
|
|
|
|
"float",
|
|
|
|
|
"long",
|
|
|
|
|
"hex",
|
|
|
|
|
"oct"
|
|
|
|
|
"chr",
|
|
|
|
|
"ord",
|
|
|
|
|
"len",
|
|
|
|
|
"abs",
|
|
|
|
|
"None",
|
|
|
|
|
"True",
|
|
|
|
|
"False"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
namespace PythonEditor {
|
2013-08-23 13:41:17 +02:00
|
|
|
namespace Internal {
|
2013-01-14 23:11:10 +04:00
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
const char PythonRunConfigurationPrefix[] = "PythonEditor.RunConfiguration.";
|
|
|
|
|
const char InterpreterKey[] = "PythonEditor.RunConfiguation.Interpreter";
|
|
|
|
|
const char MainScriptKey[] = "PythonEditor.RunConfiguation.MainScript";
|
|
|
|
|
const char PythonMimeType[] = "text/x-python-project"; // ### FIXME
|
|
|
|
|
const char PythonProjectId[] = "PythonProject";
|
|
|
|
|
const char PythonProjectContext[] = "PythonProjectContext";
|
|
|
|
|
|
|
|
|
|
class PythonRunConfiguration;
|
|
|
|
|
class PythonProjectFile;
|
|
|
|
|
class PythonProject;
|
|
|
|
|
|
|
|
|
|
static QString scriptFromId(Core::Id id)
|
|
|
|
|
{
|
|
|
|
|
return id.suffixAfter(PythonRunConfigurationPrefix);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static Core::Id idFromScript(const QString &target)
|
|
|
|
|
{
|
|
|
|
|
return Core::Id(PythonRunConfigurationPrefix).withSuffix(target);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class PythonProjectManager : public IProjectManager
|
|
|
|
|
{
|
2015-07-23 14:31:43 +03:00
|
|
|
Q_OBJECT
|
2015-06-09 15:19:54 +02:00
|
|
|
public:
|
|
|
|
|
PythonProjectManager() {}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QString mimeType() const override { return QLatin1String(PythonMimeType); }
|
|
|
|
|
Project *openProject(const QString &fileName, QString *errorString) override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
void registerProject(PythonProject *project) { m_projects.append(project); }
|
|
|
|
|
void unregisterProject(PythonProject *project) { m_projects.removeAll(project); }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QList<PythonProject *> m_projects;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class PythonProject : public Project
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
PythonProject(PythonProjectManager *manager, const QString &filename);
|
2016-01-07 17:58:34 +01:00
|
|
|
~PythonProject() override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QString displayName() const override { return m_projectName; }
|
2016-01-08 11:31:06 +01:00
|
|
|
PythonProjectManager *projectManager() const override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QStringList files(FilesMode) const override { return m_files; }
|
2015-06-09 15:19:54 +02:00
|
|
|
QStringList files() const { return m_files; }
|
|
|
|
|
|
|
|
|
|
bool addFiles(const QStringList &filePaths);
|
|
|
|
|
bool removeFiles(const QStringList &filePaths);
|
|
|
|
|
bool setFiles(const QStringList &filePaths);
|
|
|
|
|
bool renameFile(const QString &filePath, const QString &newFilePath);
|
|
|
|
|
void refresh();
|
|
|
|
|
|
|
|
|
|
private:
|
2015-07-23 11:23:52 +02:00
|
|
|
RestoreResult fromMap(const QVariantMap &map, QString *errorMessage) override;
|
|
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
bool saveRawFileList(const QStringList &rawFileList);
|
|
|
|
|
bool saveRawList(const QStringList &rawList, const QString &fileName);
|
|
|
|
|
void parseProject();
|
|
|
|
|
QStringList processEntries(const QStringList &paths,
|
|
|
|
|
QHash<QString, QString> *map = 0) const;
|
|
|
|
|
|
|
|
|
|
QString m_projectName;
|
|
|
|
|
QStringList m_rawFileList;
|
|
|
|
|
QStringList m_files;
|
|
|
|
|
QHash<QString, QString> m_rawListEntries;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class PythonProjectFile : public Core::IDocument
|
|
|
|
|
{
|
|
|
|
|
public:
|
2016-01-08 11:09:37 +01:00
|
|
|
PythonProjectFile(PythonProject *parent, QString fileName) : m_project(parent)
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
setId("Generic.ProjectFile");
|
|
|
|
|
setMimeType(QLatin1String(PythonMimeType));
|
|
|
|
|
setFilePath(FileName::fromString(fileName));
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
ReloadBehavior reloadBehavior(ChangeTrigger state, ChangeType type) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(state)
|
|
|
|
|
Q_UNUSED(type)
|
|
|
|
|
return BehaviorSilent;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool reload(QString *errorString, ReloadFlag flag, ChangeType type) override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(errorString)
|
|
|
|
|
Q_UNUSED(flag)
|
|
|
|
|
if (type == TypePermissions)
|
|
|
|
|
return true;
|
|
|
|
|
m_project->refresh();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
PythonProject *m_project;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class PythonProjectNode : public ProjectNode
|
|
|
|
|
{
|
|
|
|
|
public:
|
2015-10-29 17:12:36 +01:00
|
|
|
PythonProjectNode(PythonProject *project);
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool showInSimpleTree() const override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QList<ProjectAction> supportedActions(Node *node) const override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-11-02 17:00:46 +01:00
|
|
|
QString addFileFilter() const override;
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool renameFile(const QString &filePath, const QString &newFilePath) override;
|
2015-06-09 15:19:54 +02:00
|
|
|
void refresh(QSet<QString> oldFileList = QSet<QString>());
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
typedef QHash<QString, FolderNode *> FolderByName;
|
|
|
|
|
FolderNode *createFolderByName(const QStringList &components, int end);
|
|
|
|
|
FolderNode *findFolderByName(const QStringList &components, int end);
|
|
|
|
|
void removeEmptySubFolders(FolderNode *gparent, FolderNode *parent);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
PythonProject *m_project;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class PythonRunConfigurationWidget : public QWidget
|
|
|
|
|
{
|
2015-07-23 14:31:43 +03:00
|
|
|
Q_OBJECT
|
2015-06-09 15:19:54 +02:00
|
|
|
public:
|
|
|
|
|
PythonRunConfigurationWidget(PythonRunConfiguration *runConfiguration, QWidget *parent = 0);
|
|
|
|
|
void setInterpreter(const QString &interpreter);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
PythonRunConfiguration *m_runConfiguration;
|
|
|
|
|
DetailsWidget *m_detailsContainer;
|
|
|
|
|
FancyLineEdit *m_interpreterChooser;
|
|
|
|
|
QLabel *m_scriptLabel;
|
|
|
|
|
};
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
class PythonRunConfiguration : public ProjectExplorer::RunConfiguration
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
2015-07-23 14:31:43 +03:00
|
|
|
Q_OBJECT
|
2015-07-23 11:23:52 +02:00
|
|
|
|
|
|
|
|
Q_PROPERTY(bool supportsDebugger READ supportsDebugger)
|
|
|
|
|
Q_PROPERTY(QString interpreter READ interpreter)
|
|
|
|
|
Q_PROPERTY(QString mainScript READ mainScript)
|
|
|
|
|
Q_PROPERTY(QString arguments READ arguments)
|
|
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
public:
|
2015-07-23 11:23:52 +02:00
|
|
|
PythonRunConfiguration(ProjectExplorer::Target *parent, Core::Id id);
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QWidget *createConfigurationWidget() override;
|
|
|
|
|
QVariantMap toMap() const override;
|
|
|
|
|
bool fromMap(const QVariantMap &map) override;
|
|
|
|
|
bool isEnabled() const override { return m_enabled; }
|
|
|
|
|
QString disabledReason() const override;
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool supportsDebugger() const { return true; }
|
2015-06-09 15:19:54 +02:00
|
|
|
QString mainScript() const { return m_mainScript; }
|
2015-07-23 11:23:52 +02:00
|
|
|
QString arguments() const;
|
2015-06-09 15:19:54 +02:00
|
|
|
QString interpreter() const { return m_interpreter; }
|
|
|
|
|
void setInterpreter(const QString &interpreter) { m_interpreter = interpreter; }
|
|
|
|
|
void setEnabled(bool b);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
friend class PythonRunConfigurationFactory;
|
2015-07-23 11:23:52 +02:00
|
|
|
PythonRunConfiguration(ProjectExplorer::Target *parent, PythonRunConfiguration *source);
|
2015-06-09 15:19:54 +02:00
|
|
|
QString defaultDisplayName() const;
|
|
|
|
|
|
|
|
|
|
QString m_interpreter;
|
|
|
|
|
QString m_mainScript;
|
|
|
|
|
bool m_enabled;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class PythonRunControl : public RunControl
|
|
|
|
|
{
|
2015-07-23 14:31:43 +03:00
|
|
|
Q_OBJECT
|
2015-06-09 15:19:54 +02:00
|
|
|
public:
|
2015-06-29 10:36:29 +03:00
|
|
|
PythonRunControl(PythonRunConfiguration *runConfiguration, Core::Id mode);
|
2015-06-09 15:19:54 +02:00
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
void start() override;
|
|
|
|
|
StopResult stop() override;
|
|
|
|
|
bool isRunning() const override { return m_running; }
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void processStarted();
|
|
|
|
|
void processExited(int exitCode, QProcess::ExitStatus status);
|
|
|
|
|
void slotAppendMessage(const QString &err, Utils::OutputFormat isError);
|
|
|
|
|
|
|
|
|
|
ApplicationLauncher m_applicationLauncher;
|
|
|
|
|
QString m_interpreter;
|
|
|
|
|
QString m_mainScript;
|
|
|
|
|
QString m_commandLineArguments;
|
2016-03-11 07:32:46 +01:00
|
|
|
Utils::Environment m_environment;
|
2015-06-09 15:19:54 +02:00
|
|
|
ApplicationLauncher::Mode m_runMode;
|
|
|
|
|
bool m_running;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
PythonRunConfiguration::PythonRunConfiguration(Target *parent, Core::Id id) :
|
|
|
|
|
RunConfiguration(parent, id),
|
|
|
|
|
m_mainScript(scriptFromId(id)),
|
|
|
|
|
m_enabled(true)
|
|
|
|
|
{
|
|
|
|
|
Environment sysEnv = Environment::systemEnvironment();
|
|
|
|
|
const QString exec = sysEnv.searchInPath(QLatin1String("python")).toString();
|
|
|
|
|
m_interpreter = exec.isEmpty() ? QLatin1String("python") : exec;
|
|
|
|
|
|
2016-01-25 15:00:20 +01:00
|
|
|
addExtraAspect(new LocalEnvironmentAspect(this, LocalEnvironmentAspect::BaseEnvironmentModifier()));
|
2015-06-09 15:19:54 +02:00
|
|
|
addExtraAspect(new ArgumentsAspect(this, QStringLiteral("PythonEditor.RunConfiguration.Arguments")));
|
|
|
|
|
addExtraAspect(new TerminalAspect(this, QStringLiteral("PythonEditor.RunConfiguration.UseTerminal")));
|
|
|
|
|
setDefaultDisplayName(defaultDisplayName());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PythonRunConfiguration::PythonRunConfiguration(Target *parent, PythonRunConfiguration *source) :
|
|
|
|
|
RunConfiguration(parent, source),
|
|
|
|
|
m_interpreter(source->interpreter()),
|
|
|
|
|
m_mainScript(source->m_mainScript),
|
|
|
|
|
m_enabled(source->m_enabled)
|
|
|
|
|
{
|
|
|
|
|
setDefaultDisplayName(defaultDisplayName());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QVariantMap PythonRunConfiguration::toMap() const
|
|
|
|
|
{
|
|
|
|
|
QVariantMap map(RunConfiguration::toMap());
|
|
|
|
|
map.insert(QLatin1String(MainScriptKey), m_mainScript);
|
|
|
|
|
map.insert(QLatin1String(InterpreterKey), m_interpreter);
|
|
|
|
|
return map;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonRunConfiguration::fromMap(const QVariantMap &map)
|
|
|
|
|
{
|
|
|
|
|
m_mainScript = map.value(QLatin1String(MainScriptKey)).toString();
|
|
|
|
|
m_interpreter = map.value(QLatin1String(InterpreterKey)).toString();
|
|
|
|
|
return RunConfiguration::fromMap(map);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PythonRunConfiguration::defaultDisplayName() const
|
|
|
|
|
{
|
|
|
|
|
QString result = tr("Run %1").arg(m_mainScript);
|
|
|
|
|
if (!m_enabled) {
|
|
|
|
|
result += QLatin1Char(' ');
|
|
|
|
|
result += tr("(disabled)");
|
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QWidget *PythonRunConfiguration::createConfigurationWidget()
|
|
|
|
|
{
|
|
|
|
|
return new PythonRunConfigurationWidget(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunConfiguration::setEnabled(bool b)
|
|
|
|
|
{
|
|
|
|
|
if (m_enabled == b)
|
|
|
|
|
return;
|
|
|
|
|
m_enabled = b;
|
|
|
|
|
emit enabledChanged();
|
|
|
|
|
setDefaultDisplayName(defaultDisplayName());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString PythonRunConfiguration::disabledReason() const
|
|
|
|
|
{
|
|
|
|
|
if (!m_enabled)
|
|
|
|
|
return tr("The script is currently disabled.");
|
|
|
|
|
return QString();
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QString PythonRunConfiguration::arguments() const
|
|
|
|
|
{
|
|
|
|
|
auto aspect = extraAspect<ArgumentsAspect>();
|
|
|
|
|
QTC_ASSERT(aspect, return QString());
|
|
|
|
|
return aspect->arguments();
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
PythonRunConfigurationWidget::PythonRunConfigurationWidget(PythonRunConfiguration *runConfiguration, QWidget *parent)
|
|
|
|
|
: QWidget(parent), m_runConfiguration(runConfiguration)
|
|
|
|
|
{
|
|
|
|
|
auto fl = new QFormLayout();
|
|
|
|
|
fl->setMargin(0);
|
|
|
|
|
fl->setFieldGrowthPolicy(QFormLayout::ExpandingFieldsGrow);
|
|
|
|
|
|
|
|
|
|
m_interpreterChooser = new FancyLineEdit(this);
|
|
|
|
|
m_interpreterChooser->setText(runConfiguration->interpreter());
|
|
|
|
|
connect(m_interpreterChooser, &QLineEdit::textChanged,
|
|
|
|
|
this, &PythonRunConfigurationWidget::setInterpreter);
|
|
|
|
|
|
|
|
|
|
m_scriptLabel = new QLabel(this);
|
|
|
|
|
m_scriptLabel->setText(runConfiguration->mainScript());
|
|
|
|
|
|
|
|
|
|
fl->addRow(tr("Interpreter: "), m_interpreterChooser);
|
|
|
|
|
fl->addRow(tr("Script: "), m_scriptLabel);
|
|
|
|
|
runConfiguration->extraAspect<ArgumentsAspect>()->addToMainConfigurationWidget(this, fl);
|
|
|
|
|
runConfiguration->extraAspect<TerminalAspect>()->addToMainConfigurationWidget(this, fl);
|
|
|
|
|
|
|
|
|
|
m_detailsContainer = new DetailsWidget(this);
|
|
|
|
|
m_detailsContainer->setState(DetailsWidget::NoSummary);
|
|
|
|
|
|
|
|
|
|
auto details = new QWidget(m_detailsContainer);
|
|
|
|
|
m_detailsContainer->setWidget(details);
|
|
|
|
|
details->setLayout(fl);
|
|
|
|
|
|
|
|
|
|
auto vbx = new QVBoxLayout(this);
|
|
|
|
|
vbx->setMargin(0);
|
|
|
|
|
vbx->addWidget(m_detailsContainer);
|
|
|
|
|
|
|
|
|
|
setEnabled(runConfiguration->isEnabled());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Project *PythonProjectManager::openProject(const QString &fileName, QString *errorString)
|
|
|
|
|
{
|
|
|
|
|
if (!QFileInfo(fileName).isFile()) {
|
|
|
|
|
if (errorString)
|
|
|
|
|
*errorString = tr("Failed opening project \"%1\": Project is not a file.")
|
|
|
|
|
.arg(fileName);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return new PythonProject(this, fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class PythonRunConfigurationFactory : public IRunConfigurationFactory
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
PythonRunConfigurationFactory()
|
|
|
|
|
{
|
|
|
|
|
setObjectName(QLatin1String("PythonRunConfigurationFactory"));
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QList<Core::Id> availableCreationIds(Target *parent, CreationMode mode) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(mode);
|
|
|
|
|
if (!canHandle(parent))
|
|
|
|
|
return {};
|
|
|
|
|
//return { Core::Id(PythonExecutableId) };
|
|
|
|
|
|
|
|
|
|
PythonProject *project = static_cast<PythonProject *>(parent->project());
|
|
|
|
|
QList<Core::Id> allIds;
|
|
|
|
|
foreach (const QString &file, project->files())
|
|
|
|
|
allIds.append(idFromScript(file));
|
|
|
|
|
return allIds;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
QString displayNameForId(Core::Id id) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
return scriptFromId(id);
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool canCreate(Target *parent, Core::Id id) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
if (!canHandle(parent))
|
|
|
|
|
return false;
|
|
|
|
|
PythonProject *project = static_cast<PythonProject *>(parent->project());
|
|
|
|
|
return project->files().contains(scriptFromId(id));
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool canRestore(Target *parent, const QVariantMap &map) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(parent);
|
|
|
|
|
return idFromMap(map).name().startsWith(PythonRunConfigurationPrefix);
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
bool canClone(Target *parent, RunConfiguration *source) const override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
if (!canHandle(parent))
|
|
|
|
|
return false;
|
|
|
|
|
return source->id().name().startsWith(PythonRunConfigurationPrefix);
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
RunConfiguration *clone(Target *parent, RunConfiguration *source) override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
if (!canClone(parent, source))
|
|
|
|
|
return 0;
|
|
|
|
|
return new PythonRunConfiguration(parent, static_cast<PythonRunConfiguration*>(source));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
bool canHandle(Target *parent) const { return dynamic_cast<PythonProject *>(parent->project()); }
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
RunConfiguration *doCreate(Target *parent, Core::Id id) override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
return new PythonRunConfiguration(parent, id);
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
RunConfiguration *doRestore(Target *parent, const QVariantMap &map) override
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Core::Id id(idFromMap(map));
|
|
|
|
|
return new PythonRunConfiguration(parent, id);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-01-08 12:12:27 +01:00
|
|
|
PythonProject::PythonProject(PythonProjectManager *manager, const QString &fileName)
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
setId(PythonProjectId);
|
2016-01-08 11:31:06 +01:00
|
|
|
setProjectManager(manager);
|
2016-01-08 12:12:27 +01:00
|
|
|
setDocument(new PythonProjectFile(this, fileName));
|
2016-01-08 11:09:37 +01:00
|
|
|
DocumentManager::addDocument(document());
|
2016-01-08 12:49:00 +01:00
|
|
|
setRootProjectNode(new PythonProjectNode(this));
|
2016-01-08 11:09:37 +01:00
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
setProjectContext(Context(PythonProjectContext));
|
|
|
|
|
setProjectLanguages(Context(ProjectExplorer::Constants::LANG_CXX));
|
|
|
|
|
|
2016-01-08 12:12:27 +01:00
|
|
|
QFileInfo fileInfo = projectFilePath().toFileInfo();
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
m_projectName = fileInfo.completeBaseName();
|
|
|
|
|
|
2016-01-08 11:31:06 +01:00
|
|
|
projectManager()->registerProject(this);
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PythonProject::~PythonProject()
|
|
|
|
|
{
|
2016-01-08 11:31:06 +01:00
|
|
|
projectManager()->unregisterProject(this);
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
2016-01-08 11:31:06 +01:00
|
|
|
PythonProjectManager *PythonProject::projectManager() const
|
|
|
|
|
{
|
|
|
|
|
return static_cast<PythonProjectManager *>(Project::projectManager());
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
static QStringList readLines(const QString &absoluteFileName)
|
|
|
|
|
{
|
|
|
|
|
QStringList lines;
|
|
|
|
|
|
|
|
|
|
QFile file(absoluteFileName);
|
|
|
|
|
if (file.open(QFile::ReadOnly)) {
|
|
|
|
|
QTextStream stream(&file);
|
|
|
|
|
|
|
|
|
|
forever {
|
|
|
|
|
QString line = stream.readLine();
|
|
|
|
|
if (line.isNull())
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
lines.append(line);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return lines;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::saveRawFileList(const QStringList &rawFileList)
|
|
|
|
|
{
|
2016-01-08 12:12:27 +01:00
|
|
|
bool result = saveRawList(rawFileList, projectFilePath().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
// refresh(PythonProject::Files);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::saveRawList(const QStringList &rawList, const QString &fileName)
|
|
|
|
|
{
|
|
|
|
|
DocumentManager::expectFileChange(fileName);
|
|
|
|
|
// Make sure we can open the file for writing
|
|
|
|
|
FileSaver saver(fileName, QIODevice::Text);
|
|
|
|
|
if (!saver.hasError()) {
|
|
|
|
|
QTextStream stream(saver.file());
|
|
|
|
|
foreach (const QString &filePath, rawList)
|
|
|
|
|
stream << filePath << QLatin1Char('\n');
|
|
|
|
|
saver.setResult(&stream);
|
|
|
|
|
}
|
|
|
|
|
bool result = saver.finalize(ICore::mainWindow());
|
|
|
|
|
DocumentManager::unexpectFileChange(fileName);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::addFiles(const QStringList &filePaths)
|
|
|
|
|
{
|
|
|
|
|
QStringList newList = m_rawFileList;
|
|
|
|
|
|
2016-01-08 12:12:27 +01:00
|
|
|
QDir baseDir(projectDirectory().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
foreach (const QString &filePath, filePaths)
|
|
|
|
|
newList.append(baseDir.relativeFilePath(filePath));
|
|
|
|
|
|
|
|
|
|
QSet<QString> toAdd;
|
|
|
|
|
|
|
|
|
|
foreach (const QString &filePath, filePaths) {
|
|
|
|
|
QString directory = QFileInfo(filePath).absolutePath();
|
|
|
|
|
if (!toAdd.contains(directory))
|
|
|
|
|
toAdd << directory;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-08 12:12:27 +01:00
|
|
|
bool result = saveRawList(newList, projectFilePath().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
refresh();
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::removeFiles(const QStringList &filePaths)
|
|
|
|
|
{
|
|
|
|
|
QStringList newList = m_rawFileList;
|
|
|
|
|
|
|
|
|
|
foreach (const QString &filePath, filePaths) {
|
|
|
|
|
QHash<QString, QString>::iterator i = m_rawListEntries.find(filePath);
|
|
|
|
|
if (i != m_rawListEntries.end())
|
|
|
|
|
newList.removeOne(i.value());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return saveRawFileList(newList);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::setFiles(const QStringList &filePaths)
|
|
|
|
|
{
|
|
|
|
|
QStringList newList;
|
2016-01-08 12:12:27 +01:00
|
|
|
QDir baseDir(projectFilePath().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
foreach (const QString &filePath, filePaths)
|
|
|
|
|
newList.append(baseDir.relativeFilePath(filePath));
|
|
|
|
|
|
|
|
|
|
return saveRawFileList(newList);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProject::renameFile(const QString &filePath, const QString &newFilePath)
|
|
|
|
|
{
|
|
|
|
|
QStringList newList = m_rawFileList;
|
|
|
|
|
|
|
|
|
|
QHash<QString, QString>::iterator i = m_rawListEntries.find(filePath);
|
|
|
|
|
if (i != m_rawListEntries.end()) {
|
|
|
|
|
int index = newList.indexOf(i.value());
|
|
|
|
|
if (index != -1) {
|
2016-01-08 12:12:27 +01:00
|
|
|
QDir baseDir(projectFilePath().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
newList.replace(index, baseDir.relativeFilePath(newFilePath));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return saveRawFileList(newList);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonProject::parseProject()
|
|
|
|
|
{
|
|
|
|
|
m_rawListEntries.clear();
|
2016-01-08 12:12:27 +01:00
|
|
|
m_rawFileList = readLines(projectFilePath().toString());
|
|
|
|
|
m_rawFileList << projectFilePath().fileName();
|
2015-06-09 15:19:54 +02:00
|
|
|
m_files = processEntries(m_rawFileList, &m_rawListEntries);
|
|
|
|
|
emit fileListChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief Provides displayName relative to project node
|
|
|
|
|
*/
|
|
|
|
|
class PythonFileNode : public ProjectExplorer::FileNode
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
PythonFileNode(const Utils::FileName &filePath, const QString &nodeDisplayName)
|
|
|
|
|
: ProjectExplorer::FileNode(filePath, SourceType, false)
|
|
|
|
|
, m_displayName(nodeDisplayName)
|
|
|
|
|
{}
|
|
|
|
|
|
2015-10-29 16:59:27 +01:00
|
|
|
QString displayName() const override { return m_displayName; }
|
2015-06-09 15:19:54 +02:00
|
|
|
private:
|
|
|
|
|
QString m_displayName;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void PythonProject::refresh()
|
|
|
|
|
{
|
2016-01-08 12:49:00 +01:00
|
|
|
rootProjectNode()->removeFileNodes(rootProjectNode()->fileNodes());
|
2015-06-09 15:19:54 +02:00
|
|
|
parseProject();
|
|
|
|
|
|
2016-01-08 12:12:27 +01:00
|
|
|
QDir baseDir(projectDirectory().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
QList<FileNode *> fileNodes;
|
|
|
|
|
foreach (const QString &file, m_files) {
|
|
|
|
|
QString displayName = baseDir.relativeFilePath(file);
|
|
|
|
|
fileNodes.append(new PythonFileNode(FileName::fromString(file), displayName));
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-08 12:49:00 +01:00
|
|
|
rootProjectNode()->addFileNodes(fileNodes);
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Expands environment variables in the given \a string when they are written
|
|
|
|
|
* like $$(VARIABLE).
|
|
|
|
|
*/
|
|
|
|
|
static void expandEnvironmentVariables(const QProcessEnvironment &env, QString &string)
|
|
|
|
|
{
|
|
|
|
|
static QRegExp candidate(QLatin1String("\\$\\$\\((.+)\\)"));
|
|
|
|
|
|
|
|
|
|
int index = candidate.indexIn(string);
|
|
|
|
|
while (index != -1) {
|
|
|
|
|
const QString value = env.value(candidate.cap(1));
|
|
|
|
|
|
|
|
|
|
string.replace(index, candidate.matchedLength(), value);
|
|
|
|
|
index += value.length();
|
|
|
|
|
|
|
|
|
|
index = candidate.indexIn(string, index);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Expands environment variables and converts the path from relative to the
|
|
|
|
|
* project to an absolute path.
|
|
|
|
|
*
|
|
|
|
|
* The \a map variable is an optional argument that will map the returned
|
|
|
|
|
* absolute paths back to their original \a entries.
|
|
|
|
|
*/
|
|
|
|
|
QStringList PythonProject::processEntries(const QStringList &paths,
|
|
|
|
|
QHash<QString, QString> *map) const
|
|
|
|
|
{
|
|
|
|
|
const QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
|
2016-01-08 12:12:27 +01:00
|
|
|
const QDir projectDir(projectDirectory().toString());
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
QFileInfo fileInfo;
|
|
|
|
|
QStringList absolutePaths;
|
|
|
|
|
foreach (const QString &path, paths) {
|
|
|
|
|
QString trimmedPath = path.trimmed();
|
|
|
|
|
if (trimmedPath.isEmpty())
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
expandEnvironmentVariables(env, trimmedPath);
|
|
|
|
|
|
|
|
|
|
trimmedPath = FileName::fromUserInput(trimmedPath).toString();
|
|
|
|
|
|
|
|
|
|
fileInfo.setFile(projectDir, trimmedPath);
|
|
|
|
|
if (fileInfo.exists()) {
|
|
|
|
|
const QString absPath = fileInfo.absoluteFilePath();
|
|
|
|
|
absolutePaths.append(absPath);
|
|
|
|
|
if (map)
|
|
|
|
|
map->insert(absPath, trimmedPath);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
absolutePaths.removeDuplicates();
|
|
|
|
|
return absolutePaths;
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
Project::RestoreResult PythonProject::fromMap(const QVariantMap &map, QString *errorMessage)
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
2015-07-23 11:23:52 +02:00
|
|
|
Project::RestoreResult res = Project::fromMap(map, errorMessage);
|
|
|
|
|
if (res == RestoreResult::Ok) {
|
|
|
|
|
Kit *defaultKit = KitManager::defaultKit();
|
|
|
|
|
if (!activeTarget() && defaultKit)
|
2015-11-16 17:10:55 +01:00
|
|
|
addTarget(createTarget(defaultKit));
|
2015-07-23 11:23:52 +02:00
|
|
|
|
|
|
|
|
refresh();
|
|
|
|
|
|
|
|
|
|
QList<Target *> targetList = targets();
|
|
|
|
|
foreach (Target *t, targetList) {
|
2015-09-02 09:57:59 +02:00
|
|
|
const QList<RunConfiguration *> runConfigs = t->runConfigurations();
|
|
|
|
|
foreach (const QString &file, m_files) {
|
2015-09-02 09:58:52 +02:00
|
|
|
// skip the 'project' file
|
|
|
|
|
if (file.endsWith(QLatin1String(".pyqtc")))
|
|
|
|
|
continue;
|
2015-09-02 09:57:59 +02:00
|
|
|
const Id id = idFromScript(file);
|
|
|
|
|
bool alreadyPresent = false;
|
|
|
|
|
foreach (RunConfiguration *runCfg, runConfigs) {
|
|
|
|
|
if (runCfg->id() == id) {
|
|
|
|
|
alreadyPresent = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!alreadyPresent)
|
|
|
|
|
t->addRunConfiguration(new PythonRunConfiguration(t, id));
|
|
|
|
|
}
|
2015-07-23 11:23:52 +02:00
|
|
|
}
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
2015-07-23 11:23:52 +02:00
|
|
|
return res;
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
2015-10-29 17:12:36 +01:00
|
|
|
PythonProjectNode::PythonProjectNode(PythonProject *project)
|
|
|
|
|
: ProjectNode(project->projectFilePath())
|
2015-06-09 15:19:54 +02:00
|
|
|
, m_project(project)
|
|
|
|
|
{
|
2015-10-29 17:12:36 +01:00
|
|
|
setDisplayName(project->projectFilePath().toFileInfo().completeBaseName());
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QHash<QString, QStringList> sortFilesIntoPaths(const QString &base, const QSet<QString> &files)
|
|
|
|
|
{
|
|
|
|
|
QHash<QString, QStringList> filesInPath;
|
|
|
|
|
const QDir baseDir(base);
|
|
|
|
|
|
|
|
|
|
foreach (const QString &absoluteFileName, files) {
|
|
|
|
|
QFileInfo fileInfo(absoluteFileName);
|
|
|
|
|
FileName absoluteFilePath = FileName::fromString(fileInfo.path());
|
|
|
|
|
QString relativeFilePath;
|
|
|
|
|
|
|
|
|
|
if (absoluteFilePath.isChildOf(baseDir)) {
|
|
|
|
|
relativeFilePath = absoluteFilePath.relativeChildPath(FileName::fromString(base)).toString();
|
|
|
|
|
} else {
|
|
|
|
|
// 'file' is not part of the project.
|
|
|
|
|
relativeFilePath = baseDir.relativeFilePath(absoluteFilePath.toString());
|
|
|
|
|
if (relativeFilePath.endsWith(QLatin1Char('/')))
|
|
|
|
|
relativeFilePath.chop(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
filesInPath[relativeFilePath].append(absoluteFileName);
|
|
|
|
|
}
|
|
|
|
|
return filesInPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonProjectNode::refresh(QSet<QString> oldFileList)
|
|
|
|
|
{
|
|
|
|
|
typedef QHash<QString, QStringList> FilesInPathHash;
|
|
|
|
|
typedef FilesInPathHash::ConstIterator FilesInPathHashConstIt;
|
|
|
|
|
|
|
|
|
|
// Do those separately
|
|
|
|
|
oldFileList.remove(m_project->projectFilePath().toString());
|
|
|
|
|
|
|
|
|
|
QSet<QString> newFileList = m_project->files().toSet();
|
|
|
|
|
newFileList.remove(m_project->projectFilePath().toString());
|
|
|
|
|
|
|
|
|
|
QSet<QString> removed = oldFileList;
|
|
|
|
|
removed.subtract(newFileList);
|
|
|
|
|
QSet<QString> added = newFileList;
|
|
|
|
|
added.subtract(oldFileList);
|
|
|
|
|
|
2015-10-29 17:53:47 +01:00
|
|
|
QString baseDir = filePath().toFileInfo().absolutePath();
|
2015-06-09 15:19:54 +02:00
|
|
|
FilesInPathHash filesInPaths = sortFilesIntoPaths(baseDir, added);
|
|
|
|
|
|
|
|
|
|
FilesInPathHashConstIt cend = filesInPaths.constEnd();
|
|
|
|
|
for (FilesInPathHashConstIt it = filesInPaths.constBegin(); it != cend; ++it) {
|
|
|
|
|
const QString &filePath = it.key();
|
|
|
|
|
QStringList components;
|
|
|
|
|
if (!filePath.isEmpty())
|
|
|
|
|
components = filePath.split(QLatin1Char('/'));
|
|
|
|
|
FolderNode *folder = findFolderByName(components, components.size());
|
|
|
|
|
if (!folder)
|
|
|
|
|
folder = createFolderByName(components, components.size());
|
|
|
|
|
|
|
|
|
|
QList<FileNode *> fileNodes;
|
|
|
|
|
foreach (const QString &file, it.value()) {
|
|
|
|
|
FileType fileType = SourceType; // ### FIXME
|
|
|
|
|
if (file.endsWith(QLatin1String(".qrc")))
|
|
|
|
|
fileType = ResourceType;
|
|
|
|
|
FileNode *fileNode = new FileNode(FileName::fromString(file),
|
|
|
|
|
fileType, /*generated = */ false);
|
|
|
|
|
fileNodes.append(fileNode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
folder->addFileNodes(fileNodes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
filesInPaths = sortFilesIntoPaths(baseDir, removed);
|
|
|
|
|
cend = filesInPaths.constEnd();
|
|
|
|
|
for (FilesInPathHashConstIt it = filesInPaths.constBegin(); it != cend; ++it) {
|
|
|
|
|
const QString &filePath = it.key();
|
|
|
|
|
QStringList components;
|
|
|
|
|
if (!filePath.isEmpty())
|
|
|
|
|
components = filePath.split(QLatin1Char('/'));
|
|
|
|
|
FolderNode *folder = findFolderByName(components, components.size());
|
|
|
|
|
|
|
|
|
|
QList<FileNode *> fileNodes;
|
|
|
|
|
foreach (const QString &file, it.value()) {
|
|
|
|
|
foreach (FileNode *fn, folder->fileNodes()) {
|
2015-10-29 17:53:47 +01:00
|
|
|
if (fn->filePath().toString() == file)
|
2015-06-09 15:19:54 +02:00
|
|
|
fileNodes.append(fn);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
folder->removeFileNodes(fileNodes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach (FolderNode *fn, subFolderNodes())
|
|
|
|
|
removeEmptySubFolders(this, fn);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonProjectNode::removeEmptySubFolders(FolderNode *gparent, FolderNode *parent)
|
|
|
|
|
{
|
|
|
|
|
foreach (FolderNode *fn, parent->subFolderNodes())
|
|
|
|
|
removeEmptySubFolders(parent, fn);
|
|
|
|
|
|
|
|
|
|
if (parent->subFolderNodes().isEmpty() && parent->fileNodes().isEmpty())
|
|
|
|
|
gparent->removeFolderNodes(QList<FolderNode*>() << parent);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FolderNode *PythonProjectNode::createFolderByName(const QStringList &components, int end)
|
|
|
|
|
{
|
|
|
|
|
if (end == 0)
|
|
|
|
|
return this;
|
|
|
|
|
|
|
|
|
|
QString folderName;
|
|
|
|
|
for (int i = 0; i < end; ++i) {
|
|
|
|
|
folderName.append(components.at(i));
|
|
|
|
|
folderName += QLatin1Char('/');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const QString component = components.at(end - 1);
|
|
|
|
|
|
2015-10-29 17:53:47 +01:00
|
|
|
const FileName folderPath = filePath().parentDir().appendPath(folderName);
|
2015-06-09 15:19:54 +02:00
|
|
|
FolderNode *folder = new FolderNode(folderPath);
|
|
|
|
|
folder->setDisplayName(component);
|
|
|
|
|
|
|
|
|
|
FolderNode *parent = findFolderByName(components, end - 1);
|
|
|
|
|
if (!parent)
|
|
|
|
|
parent = createFolderByName(components, end - 1);
|
|
|
|
|
parent->addFolderNodes(QList<FolderNode*>() << folder);
|
|
|
|
|
|
|
|
|
|
return folder;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FolderNode *PythonProjectNode::findFolderByName(const QStringList &components, int end)
|
|
|
|
|
{
|
|
|
|
|
if (end == 0)
|
|
|
|
|
return this;
|
|
|
|
|
|
|
|
|
|
QString folderName;
|
|
|
|
|
for (int i = 0; i < end; ++i) {
|
|
|
|
|
folderName.append(components.at(i));
|
|
|
|
|
folderName += QLatin1Char('/');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
FolderNode *parent = findFolderByName(components, end - 1);
|
|
|
|
|
|
|
|
|
|
if (!parent)
|
|
|
|
|
return 0;
|
|
|
|
|
|
2015-10-29 17:53:47 +01:00
|
|
|
const QString baseDir = filePath().toFileInfo().path();
|
2015-06-09 15:19:54 +02:00
|
|
|
foreach (FolderNode *fn, parent->subFolderNodes()) {
|
2015-10-29 17:53:47 +01:00
|
|
|
if (fn->filePath().toString() == baseDir + QLatin1Char('/') + folderName)
|
2015-06-09 15:19:54 +02:00
|
|
|
return fn;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool PythonProjectNode::showInSimpleTree() const
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QList<ProjectAction> PythonProjectNode::supportedActions(Node *node) const
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(node);
|
|
|
|
|
//return { AddNewFile, AddExistingFile, AddExistingDirectory, RemoveFile, Rename };
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2015-11-02 17:00:46 +01:00
|
|
|
QString PythonProjectNode::addFileFilter() const
|
|
|
|
|
{
|
|
|
|
|
return QLatin1String("*.py");
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
bool PythonProjectNode::renameFile(const QString &filePath, const QString &newFilePath)
|
|
|
|
|
{
|
|
|
|
|
return m_project->renameFile(filePath, newFilePath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PythonRunControlFactory
|
|
|
|
|
|
|
|
|
|
class PythonRunControlFactory : public IRunControlFactory
|
|
|
|
|
{
|
|
|
|
|
public:
|
2016-01-29 12:28:31 +01:00
|
|
|
bool canRun(RunConfiguration *runConfiguration, Core::Id mode) const override;
|
|
|
|
|
RunControl *create(RunConfiguration *runConfiguration, Core::Id mode, QString *errorMessage) override;
|
2015-06-09 15:19:54 +02:00
|
|
|
};
|
|
|
|
|
|
2015-06-29 10:36:29 +03:00
|
|
|
bool PythonRunControlFactory::canRun(RunConfiguration *runConfiguration, Core::Id mode) const
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
2015-07-23 11:23:52 +02:00
|
|
|
return mode == ProjectExplorer::Constants::NORMAL_RUN_MODE
|
|
|
|
|
&& dynamic_cast<PythonRunConfiguration *>(runConfiguration);
|
2015-06-09 15:19:54 +02:00
|
|
|
}
|
|
|
|
|
|
2015-06-29 10:36:29 +03:00
|
|
|
RunControl *PythonRunControlFactory::create(RunConfiguration *runConfiguration, Core::Id mode, QString *errorMessage)
|
2015-06-09 15:19:54 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(errorMessage)
|
|
|
|
|
QTC_ASSERT(canRun(runConfiguration, mode), return 0);
|
|
|
|
|
return new PythonRunControl(static_cast<PythonRunConfiguration *>(runConfiguration), mode);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PythonRunControl
|
|
|
|
|
|
2015-06-29 10:36:29 +03:00
|
|
|
PythonRunControl::PythonRunControl(PythonRunConfiguration *rc, Core::Id mode)
|
2015-06-09 15:19:54 +02:00
|
|
|
: RunControl(rc, mode), m_running(false)
|
|
|
|
|
{
|
2015-11-23 16:41:54 +01:00
|
|
|
setIcon(ProjectExplorer::Icons::RUN_SMALL);
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
m_interpreter = rc->interpreter();
|
|
|
|
|
m_mainScript = rc->mainScript();
|
|
|
|
|
m_runMode = rc->extraAspect<TerminalAspect>()->runMode();
|
|
|
|
|
m_commandLineArguments = rc->extraAspect<ArgumentsAspect>()->arguments();
|
2016-03-11 07:32:46 +01:00
|
|
|
m_environment = rc->extraAspect<EnvironmentAspect>()->environment();
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
connect(&m_applicationLauncher, &ApplicationLauncher::appendMessage,
|
|
|
|
|
this, &PythonRunControl::slotAppendMessage);
|
|
|
|
|
connect(&m_applicationLauncher, &ApplicationLauncher::processStarted,
|
|
|
|
|
this, &PythonRunControl::processStarted);
|
|
|
|
|
connect(&m_applicationLauncher, &ApplicationLauncher::processExited,
|
|
|
|
|
this, &PythonRunControl::processExited);
|
|
|
|
|
connect(&m_applicationLauncher, &ApplicationLauncher::bringToForegroundRequested,
|
|
|
|
|
this, &RunControl::bringApplicationToForeground);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunControl::start()
|
|
|
|
|
{
|
|
|
|
|
emit started();
|
|
|
|
|
if (m_interpreter.isEmpty()) {
|
|
|
|
|
appendMessage(tr("No Python interpreter specified.") + QLatin1Char('\n'), Utils::ErrorMessageFormat);
|
|
|
|
|
emit finished();
|
|
|
|
|
} else if (!QFileInfo::exists(m_interpreter)) {
|
|
|
|
|
appendMessage(tr("Python interpreter %1 does not exist.").arg(QDir::toNativeSeparators(m_interpreter)) + QLatin1Char('\n'),
|
|
|
|
|
Utils::ErrorMessageFormat);
|
|
|
|
|
emit finished();
|
|
|
|
|
} else {
|
|
|
|
|
m_running = true;
|
|
|
|
|
QString msg = tr("Starting %1...").arg(QDir::toNativeSeparators(m_interpreter)) + QLatin1Char('\n');
|
|
|
|
|
appendMessage(msg, Utils::NormalMessageFormat);
|
|
|
|
|
|
2016-01-27 16:56:36 +01:00
|
|
|
StandardRunnable r;
|
|
|
|
|
QtcProcess::addArg(&r.commandLineArguments, m_mainScript);
|
|
|
|
|
QtcProcess::addArgs(&r.commandLineArguments, m_commandLineArguments);
|
|
|
|
|
r.executable = m_interpreter;
|
|
|
|
|
r.runMode = m_runMode;
|
2016-03-11 07:32:46 +01:00
|
|
|
r.environment = m_environment;
|
2016-01-27 16:56:36 +01:00
|
|
|
m_applicationLauncher.start(r);
|
2015-06-09 15:19:54 +02:00
|
|
|
|
|
|
|
|
setApplicationProcessHandle(ProcessHandle(m_applicationLauncher.applicationPID()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PythonRunControl::StopResult PythonRunControl::stop()
|
|
|
|
|
{
|
|
|
|
|
m_applicationLauncher.stop();
|
|
|
|
|
return StoppedSynchronously;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunControl::slotAppendMessage(const QString &err, Utils::OutputFormat format)
|
|
|
|
|
{
|
|
|
|
|
appendMessage(err, format);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunControl::processStarted()
|
|
|
|
|
{
|
|
|
|
|
// Console processes only know their pid after being started
|
|
|
|
|
setApplicationProcessHandle(ProcessHandle(m_applicationLauncher.applicationPID()));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunControl::processExited(int exitCode, QProcess::ExitStatus status)
|
|
|
|
|
{
|
|
|
|
|
m_running = false;
|
|
|
|
|
setApplicationProcessHandle(ProcessHandle());
|
|
|
|
|
QString msg;
|
|
|
|
|
if (status == QProcess::CrashExit) {
|
|
|
|
|
msg = tr("%1 crashed")
|
|
|
|
|
.arg(QDir::toNativeSeparators(m_interpreter));
|
|
|
|
|
} else {
|
|
|
|
|
msg = tr("%1 exited with code %2")
|
|
|
|
|
.arg(QDir::toNativeSeparators(m_interpreter)).arg(exitCode);
|
|
|
|
|
}
|
|
|
|
|
appendMessage(msg + QLatin1Char('\n'), Utils::NormalMessageFormat);
|
|
|
|
|
emit finished();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PythonRunConfigurationWidget::setInterpreter(const QString &interpreter)
|
|
|
|
|
{
|
|
|
|
|
m_runConfiguration->setInterpreter(interpreter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// PythonEditorPlugin
|
|
|
|
|
//
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2014-08-20 15:05:32 +02:00
|
|
|
static PythonEditorPlugin *m_instance = 0;
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
/// Copies identifiers from array to QSet
|
|
|
|
|
static void copyIdentifiers(const char * const words[], size_t bytesCount, QSet<QString> &result)
|
|
|
|
|
{
|
|
|
|
|
const size_t count = bytesCount / sizeof(const char * const);
|
|
|
|
|
for (size_t i = 0; i < count; ++i)
|
|
|
|
|
result.insert(QLatin1String(words[i]));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PythonEditorPlugin::PythonEditorPlugin()
|
|
|
|
|
{
|
|
|
|
|
m_instance = this;
|
|
|
|
|
copyIdentifiers(LIST_OF_PYTHON_KEYWORDS, sizeof(LIST_OF_PYTHON_KEYWORDS), m_keywords);
|
|
|
|
|
copyIdentifiers(LIST_OF_PYTHON_MAGICS, sizeof(LIST_OF_PYTHON_MAGICS), m_magics);
|
|
|
|
|
copyIdentifiers(LIST_OF_PYTHON_BUILTINS, sizeof(LIST_OF_PYTHON_BUILTINS), m_builtins);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
PythonEditorPlugin::~PythonEditorPlugin()
|
|
|
|
|
{
|
|
|
|
|
m_instance = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2013-08-30 16:38:57 +02:00
|
|
|
bool PythonEditorPlugin::initialize(const QStringList &arguments, QString *errorMessage)
|
2013-01-14 23:11:10 +04:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(arguments)
|
2015-02-04 09:32:46 +01:00
|
|
|
Q_UNUSED(errorMessage)
|
2013-01-14 23:11:10 +04:00
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
MimeDatabase::addMimeTypes(QLatin1String(":/pythoneditor/PythonEditor.mimetypes.xml"));
|
2013-01-14 23:11:10 +04:00
|
|
|
|
2015-06-09 15:19:54 +02:00
|
|
|
addAutoReleasedObject(new PythonProjectManager);
|
2014-08-20 15:05:32 +02:00
|
|
|
addAutoReleasedObject(new PythonEditorFactory);
|
2015-06-09 15:19:54 +02:00
|
|
|
addAutoReleasedObject(new PythonRunConfigurationFactory);
|
|
|
|
|
addAutoReleasedObject(new PythonRunControlFactory);
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
// Initialize editor actions handler
|
|
|
|
|
// Add MIME overlay icons (these icons displayed at Project dock panel)
|
|
|
|
|
const QIcon icon = QIcon::fromTheme(QLatin1String(C_PY_MIME_ICON));
|
2013-09-12 16:06:33 +02:00
|
|
|
if (!icon.isNull())
|
|
|
|
|
Core::FileIconProvider::registerIconOverlayForMimeType(icon, C_PY_MIMETYPE);
|
2013-01-14 23:11:10 +04:00
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QSet<QString> PythonEditorPlugin::keywords()
|
|
|
|
|
{
|
2014-08-20 15:05:32 +02:00
|
|
|
return m_instance->m_keywords;
|
2013-01-14 23:11:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QSet<QString> PythonEditorPlugin::magics()
|
|
|
|
|
{
|
2014-08-20 15:05:32 +02:00
|
|
|
return m_instance->m_magics;
|
2013-01-14 23:11:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QSet<QString> PythonEditorPlugin::builtins()
|
|
|
|
|
{
|
2014-08-20 15:05:32 +02:00
|
|
|
return m_instance->m_builtins;
|
2013-01-14 23:11:10 +04:00
|
|
|
}
|
|
|
|
|
|
2013-08-23 13:41:17 +02:00
|
|
|
} // namespace Internal
|
2013-01-14 23:11:10 +04:00
|
|
|
} // namespace PythonEditor
|
2015-07-23 14:31:43 +03:00
|
|
|
|
|
|
|
|
#include "pythoneditorplugin.moc"
|