forked from qt-creator/qt-creator
- configure with QTC_USE_NEW_MIMEDATABASE to switch to the new one in utils/mimetypes2/ - added utils/mimeutils.h header for the Qt Creator specific static wrappers, that also includes the "public" headers for MimeType et al from the new or old implementation, depending on configuration - change all utils/mimetypes/ includes to utils/mimeutils.h - move the implementation for the wrappers to utils/mimetypes(2)/mimeutils.cpp - also move the MimeDatabase declaration in the "old" implementation back to utils/mimetypes/mimedatabase.h Change-Id: Ie8de229c035d6cd9a5e4739dc0fa78d9c17228e3 Reviewed-by: Jarek Kobus <jaroslaw.kobus@qt.io> Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org> Reviewed-by: <github-actions-qt-creator@cristianadam.eu>
325 lines
14 KiB
C++
325 lines
14 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and 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.
|
|
**
|
|
** 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.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "qmljsmodelmanager.h"
|
|
#include "qmljstoolsconstants.h"
|
|
#include "qmljssemanticinfo.h"
|
|
#include "qmljsbundleprovider.h"
|
|
|
|
#include <coreplugin/icore.h>
|
|
#include <coreplugin/editormanager/documentmodel.h>
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
#include <coreplugin/messagemanager.h>
|
|
#include <coreplugin/progressmanager/progressmanager.h>
|
|
|
|
#include <cppeditor/cppmodelmanager.h>
|
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
#include <projectexplorer/buildsystem.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/projectnodes.h>
|
|
#include <projectexplorer/projecttree.h>
|
|
#include <projectexplorer/runconfiguration.h>
|
|
#include <projectexplorer/session.h>
|
|
#include <projectexplorer/target.h>
|
|
|
|
#include <qmljs/qmljsbind.h>
|
|
#include <qmljs/qmljsfindexportedcpptypes.h>
|
|
#include <qmljs/qmljsplugindumper.h>
|
|
#include <qtsupport/qtkitinformation.h>
|
|
#include <qtsupport/qtsupportconstants.h>
|
|
#include <texteditor/textdocument.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/hostosinfo.h>
|
|
#include <utils/mimeutils.h>
|
|
|
|
#include <QDir>
|
|
#include <QFile>
|
|
#include <QFileInfo>
|
|
#include <utils/runextensions.h>
|
|
#include <QTextDocument>
|
|
#include <QTextStream>
|
|
#include <QTimer>
|
|
#include <QSet>
|
|
#include <QString>
|
|
#include <QLibraryInfo>
|
|
#include <qglobal.h>
|
|
|
|
using namespace Utils;
|
|
using namespace Core;
|
|
using namespace ProjectExplorer;
|
|
using namespace QmlJS;
|
|
|
|
namespace QmlJSTools {
|
|
namespace Internal {
|
|
|
|
static void setupProjectInfoQmlBundles(ModelManagerInterface::ProjectInfo &projectInfo)
|
|
{
|
|
Target *activeTarget = nullptr;
|
|
if (projectInfo.project)
|
|
activeTarget = projectInfo.project->activeTarget();
|
|
Kit *activeKit = activeTarget ? activeTarget->kit() : KitManager::defaultKit();
|
|
const QHash<QString, QString> replacements = {{QLatin1String("$(QT_INSTALL_QML)"), projectInfo.qtQmlPath.toString()}};
|
|
|
|
for (IBundleProvider *bp : IBundleProvider::allBundleProviders())
|
|
bp->mergeBundlesForKit(activeKit, projectInfo.activeBundle, replacements);
|
|
|
|
projectInfo.extendedBundle = projectInfo.activeBundle;
|
|
|
|
if (projectInfo.project) {
|
|
QSet<Kit *> currentKits;
|
|
foreach (const Target *t, projectInfo.project->targets())
|
|
currentKits.insert(t->kit());
|
|
currentKits.remove(activeKit);
|
|
foreach (Kit *kit, currentKits) {
|
|
for (IBundleProvider *bp : IBundleProvider::allBundleProviders())
|
|
bp->mergeBundlesForKit(kit, projectInfo.extendedBundle, replacements);
|
|
}
|
|
}
|
|
}
|
|
|
|
ModelManagerInterface::ProjectInfo ModelManager::defaultProjectInfoForProject(
|
|
Project *project) const
|
|
{
|
|
ModelManagerInterface::ProjectInfo projectInfo;
|
|
projectInfo.project = project;
|
|
projectInfo.qmlDumpEnvironment = Utils::Environment::systemEnvironment();
|
|
Target *activeTarget = nullptr;
|
|
if (project) {
|
|
const QSet<QString> qmlTypeNames = { Constants::QML_MIMETYPE ,Constants::QBS_MIMETYPE,
|
|
Constants::QMLPROJECT_MIMETYPE,
|
|
Constants::QMLTYPES_MIMETYPE,
|
|
Constants::QMLUI_MIMETYPE };
|
|
projectInfo.sourceFiles = Utils::transform(project->files([&qmlTypeNames](const Node *n) {
|
|
if (!Project::SourceFiles(n))
|
|
return false;
|
|
const FileNode *fn = n->asFileNode();
|
|
return fn && fn->fileType() == FileType::QML
|
|
&& qmlTypeNames.contains(Utils::mimeTypeForFile(fn->filePath(),
|
|
MimeMatchMode::MatchExtension).name());
|
|
}), &FilePath::toString);
|
|
activeTarget = project->activeTarget();
|
|
}
|
|
Kit *activeKit = activeTarget ? activeTarget->kit() : KitManager::defaultKit();
|
|
QtSupport::QtVersion *qtVersion = QtSupport::QtKitAspect::qtVersion(activeKit);
|
|
|
|
projectInfo.tryQmlDump = false;
|
|
|
|
if (activeTarget) {
|
|
QDir baseDir;
|
|
auto addAppDir = [&baseDir, & projectInfo](const QString &mdir) {
|
|
auto dir = QDir::cleanPath(mdir);
|
|
if (!baseDir.path().isEmpty()) {
|
|
auto rDir = baseDir.relativeFilePath(dir);
|
|
// do not add directories outside the build directory
|
|
// this might happen for example when we think an executable path belongs to
|
|
// a bundle, and we need to remove extra directories, but that was not the case
|
|
if (rDir.split(u'/').contains(QStringLiteral(u"..")))
|
|
return;
|
|
}
|
|
if (!projectInfo.applicationDirectories.contains(dir))
|
|
projectInfo.applicationDirectories.append(dir);
|
|
};
|
|
|
|
if (BuildConfiguration *bc = activeTarget->activeBuildConfiguration()) {
|
|
// Append QML2_IMPORT_PATH if it is defined in build configuration.
|
|
// It enables qmlplugindump to correctly dump custom plugins or other dependent
|
|
// plugins that are not installed in default Qt qml installation directory.
|
|
projectInfo.qmlDumpEnvironment.appendOrSet("QML2_IMPORT_PATH", bc->environment().expandedValueForKey("QML2_IMPORT_PATH"), ":");
|
|
// Treat every target (library or application) in the build directory
|
|
|
|
QString dir = bc->buildDirectory().toString();
|
|
baseDir.setPath(QDir{dir}.absolutePath());
|
|
addAppDir(dir);
|
|
}
|
|
// Qml loads modules from the following sources
|
|
// 1. The build directory of the executable
|
|
// 2. Any QML_IMPORT_PATH (environment variable) or IMPORT_PATH (parameter to qt_add_qml_module)
|
|
// 3. The Qt import path
|
|
// For an IDE things are a bit more complicated because source files might be edited,
|
|
// and the directory of the executable might be outdated.
|
|
// Here we try to get the directory of the executable, adding all targets
|
|
const auto appTargets = activeTarget->buildSystem()->applicationTargets();
|
|
for (const auto &target : appTargets) {
|
|
if (target.targetFilePath.isEmpty())
|
|
continue;
|
|
auto dir = target.targetFilePath.parentDir();
|
|
projectInfo.applicationDirectories.append(dir.toString());
|
|
// unfortunately the build directory of the executable where cmake puts the qml
|
|
// might be different than the directory of the executable:
|
|
#if defined(Q_OS_WIN)
|
|
// On Windows systems QML type information is located one directory higher as we build
|
|
// in dedicated "debug" and "release" directories
|
|
addAppDir(
|
|
dir.parentDir().toString());
|
|
#elif defined(Q_OS_MACOS)
|
|
// On macOS and iOS when building a bundle this is not the case and
|
|
// we have to go up up to three additional directories
|
|
// (BundleName.app/Contents/MacOS or BundleName.app/Contents for iOS)
|
|
if (dir.fileName() == u"MacOS")
|
|
dir = dir.parentDir();
|
|
if (dir.fileName() == u"Contents")
|
|
dir = dir.parentDir().parentDir();
|
|
addAppDir(dir.toString());
|
|
#endif
|
|
}
|
|
}
|
|
if (qtVersion && qtVersion->isValid()) {
|
|
projectInfo.tryQmlDump = project && qtVersion->type() == QLatin1String(QtSupport::Constants::DESKTOPQT);
|
|
projectInfo.qtQmlPath = qtVersion->qmlPath();
|
|
projectInfo.qtVersionString = qtVersion->qtVersionString();
|
|
} else if (!activeKit || !activeKit->value(QtSupport::SuppliesQtQuickImportPath::id(), false).toBool()) {
|
|
projectInfo.qtQmlPath = FilePath::fromUserInput(QLibraryInfo::location(QLibraryInfo::Qml2ImportsPath));
|
|
projectInfo.qtVersionString = QLatin1String(qVersion());
|
|
}
|
|
|
|
projectInfo.qmlDumpPath.clear();
|
|
const QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(activeKit);
|
|
if (version && projectInfo.tryQmlDump) {
|
|
projectInfo.qmlDumpPath = version->qmlplugindumpFilePath();
|
|
projectInfo.qmlDumpHasRelocatableFlag = version->hasQmlDumpWithRelocatableFlag();
|
|
}
|
|
|
|
setupProjectInfoQmlBundles(projectInfo);
|
|
return projectInfo;
|
|
}
|
|
|
|
QHash<QString,Dialect> ModelManager::initLanguageForSuffix() const
|
|
{
|
|
QHash<QString,Dialect> res = ModelManagerInterface::languageForSuffix();
|
|
|
|
if (ICore::instance()) {
|
|
MimeType jsSourceTy = Utils::mimeTypeForName(Constants::JS_MIMETYPE);
|
|
foreach (const QString &suffix, jsSourceTy.suffixes())
|
|
res[suffix] = Dialect::JavaScript;
|
|
MimeType qmlSourceTy = Utils::mimeTypeForName(Constants::QML_MIMETYPE);
|
|
foreach (const QString &suffix, qmlSourceTy.suffixes())
|
|
res[suffix] = Dialect::Qml;
|
|
MimeType qbsSourceTy = Utils::mimeTypeForName(Constants::QBS_MIMETYPE);
|
|
foreach (const QString &suffix, qbsSourceTy.suffixes())
|
|
res[suffix] = Dialect::QmlQbs;
|
|
MimeType qmlProjectSourceTy = Utils::mimeTypeForName(Constants::QMLPROJECT_MIMETYPE);
|
|
foreach (const QString &suffix, qmlProjectSourceTy.suffixes())
|
|
res[suffix] = Dialect::QmlProject;
|
|
MimeType qmlUiSourceTy = Utils::mimeTypeForName(Constants::QMLUI_MIMETYPE);
|
|
foreach (const QString &suffix, qmlUiSourceTy.suffixes())
|
|
res[suffix] = Dialect::QmlQtQuick2Ui;
|
|
MimeType jsonSourceTy = Utils::mimeTypeForName(Constants::JSON_MIMETYPE);
|
|
foreach (const QString &suffix, jsonSourceTy.suffixes())
|
|
res[suffix] = Dialect::Json;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
QHash<QString,Dialect> ModelManager::languageForSuffix() const
|
|
{
|
|
static QHash<QString,Dialect> res = initLanguageForSuffix();
|
|
return res;
|
|
}
|
|
|
|
ModelManager::ModelManager()
|
|
{
|
|
qRegisterMetaType<QmlJSTools::SemanticInfo>("QmlJSTools::SemanticInfo");
|
|
loadDefaultQmlTypeDescriptions();
|
|
}
|
|
|
|
ModelManager::~ModelManager() = default;
|
|
|
|
void ModelManager::delayedInitialization()
|
|
{
|
|
CppEditor::CppModelManager *cppModelManager = CppEditor::CppModelManager::instance();
|
|
// It's important to have a direct connection here so we can prevent
|
|
// the source and AST of the cpp document being cleaned away.
|
|
connect(cppModelManager, &CppEditor::CppModelManager::documentUpdated,
|
|
this, &ModelManagerInterface::maybeQueueCppQmlTypeUpdate, Qt::DirectConnection);
|
|
|
|
connect(SessionManager::instance(), &SessionManager::projectRemoved,
|
|
this, &ModelManager::removeProjectInfo);
|
|
connect(SessionManager::instance(), &SessionManager::startupProjectChanged,
|
|
this, &ModelManager::updateDefaultProjectInfo);
|
|
|
|
ViewerContext qbsVContext;
|
|
qbsVContext.language = Dialect::QmlQbs;
|
|
qbsVContext.paths.append(ICore::resourcePath("qbs").toString());
|
|
setDefaultVContext(qbsVContext);
|
|
}
|
|
|
|
void ModelManager::loadDefaultQmlTypeDescriptions()
|
|
{
|
|
if (ICore::instance()) {
|
|
loadQmlTypeDescriptionsInternal(ICore::resourcePath().toString());
|
|
loadQmlTypeDescriptionsInternal(ICore::userResourcePath().toString());
|
|
}
|
|
}
|
|
|
|
void ModelManager::writeMessageInternal(const QString &msg) const
|
|
{
|
|
MessageManager::writeFlashing(msg);
|
|
}
|
|
|
|
ModelManagerInterface::WorkingCopy ModelManager::workingCopyInternal() const
|
|
{
|
|
WorkingCopy workingCopy;
|
|
|
|
if (!Core::ICore::instance())
|
|
return workingCopy;
|
|
|
|
foreach (IDocument *document, DocumentModel::openedDocuments()) {
|
|
const QString key = document->filePath().toString();
|
|
if (auto textDocument = qobject_cast<const TextEditor::TextDocument *>(document)) {
|
|
// TODO the language should be a property on the document, not the editor
|
|
if (DocumentModel::editorsForDocument(document).constFirst()
|
|
->context().contains(ProjectExplorer::Constants::QMLJS_LANGUAGE_ID)) {
|
|
workingCopy.insert(key, textDocument->plainText(),
|
|
textDocument->document()->revision());
|
|
}
|
|
}
|
|
}
|
|
|
|
return workingCopy;
|
|
}
|
|
|
|
void ModelManager::updateDefaultProjectInfo()
|
|
{
|
|
// needs to be performed in the ui thread
|
|
Project *currentProject = SessionManager::startupProject();
|
|
setDefaultProject(containsProject(currentProject)
|
|
? projectInfo(currentProject)
|
|
: defaultProjectInfoForProject(currentProject),
|
|
currentProject);
|
|
}
|
|
|
|
|
|
void ModelManager::addTaskInternal(const QFuture<void> &result, const QString &msg,
|
|
const char *taskId) const
|
|
{
|
|
ProgressManager::addTask(result, msg, taskId);
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace QmlJSTools
|