forked from qt-creator/qt-creator
Task-number: QTCREATORBUG-29259 Change-Id: Ib05737368c224fa28b2c9f091dfa0e210bc838af Reviewed-by: Thomas Hartmann <thomas.hartmann@qt.io>
327 lines
12 KiB
C++
327 lines
12 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "qmlprojectrunconfiguration.h"
|
|
#include "qmlmainfileaspect.h"
|
|
#include "qmlmultilanguageaspect.h"
|
|
#include "qmlproject.h"
|
|
#include "qmlprojectmanagerconstants.h"
|
|
#include "qmlprojectmanagertr.h"
|
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
#include <coreplugin/icore.h>
|
|
#include <coreplugin/idocument.h>
|
|
|
|
#include <projectexplorer/devicesupport/idevice.h>
|
|
#include <projectexplorer/environmentaspect.h>
|
|
#include <projectexplorer/kitinformation.h>
|
|
#include <projectexplorer/kitmanager.h>
|
|
#include <projectexplorer/projectexplorer.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/runconfigurationaspects.h>
|
|
#include <projectexplorer/runcontrol.h>
|
|
#include <projectexplorer/projectmanager.h>
|
|
#include <projectexplorer/target.h>
|
|
|
|
#include <qmldesignerbase/qmldesignerbaseplugin.h>
|
|
#include <qmldesignerbase/utils/qmlpuppetpaths.h>
|
|
|
|
#include <qtsupport/qtkitinformation.h>
|
|
#include <qtsupport/qtsupportconstants.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/aspects.h>
|
|
#include <utils/environment.h>
|
|
#include <utils/fileutils.h>
|
|
#include <utils/process.h>
|
|
#include <utils/winutils.h>
|
|
|
|
#include <qmljstools/qmljstoolsconstants.h>
|
|
|
|
using namespace Core;
|
|
using namespace ProjectExplorer;
|
|
using namespace QtSupport;
|
|
using namespace Utils;
|
|
|
|
namespace QmlProjectManager::Internal {
|
|
|
|
// QmlProjectRunConfiguration
|
|
|
|
class QmlProjectRunConfiguration final : public RunConfiguration
|
|
{
|
|
public:
|
|
QmlProjectRunConfiguration(Target *target, Id id);
|
|
|
|
private:
|
|
QString disabledReason() const final;
|
|
bool isEnabled() const final;
|
|
|
|
FilePath mainScript() const;
|
|
FilePath qmlRuntimeFilePath() const;
|
|
void createQtVersionAspect();
|
|
|
|
FilePathAspect *m_qmlViewerAspect = nullptr;
|
|
QmlMainFileAspect *m_qmlMainFileAspect = nullptr;
|
|
QmlMultiLanguageAspect *m_multiLanguageAspect = nullptr;
|
|
SelectionAspect *m_qtversionAspect = nullptr;
|
|
mutable bool usePuppetAsQmlRuntime = false;
|
|
};
|
|
|
|
QmlProjectRunConfiguration::QmlProjectRunConfiguration(Target *target, Id id)
|
|
: RunConfiguration(target, id)
|
|
{
|
|
m_qmlViewerAspect = addAspect<FilePathAspect>();
|
|
m_qmlViewerAspect->setLabelText(Tr::tr("Override device QML viewer:"));
|
|
m_qmlViewerAspect->setPlaceHolderText(qmlRuntimeFilePath().toUserOutput());
|
|
m_qmlViewerAspect->setHistoryCompleter("QmlProjectManager.viewer.history");
|
|
m_qmlViewerAspect->setSettingsKey(Constants::QML_VIEWER_KEY);
|
|
|
|
auto argumentAspect = addAspect<ArgumentsAspect>(macroExpander());
|
|
argumentAspect->setSettingsKey(Constants::QML_VIEWER_ARGUMENTS_KEY);
|
|
|
|
setCommandLineGetter([this, target] {
|
|
const FilePath qmlRuntime = qmlRuntimeFilePath();
|
|
CommandLine cmd(qmlRuntime);
|
|
if (usePuppetAsQmlRuntime)
|
|
cmd.addArg("--qml-runtime");
|
|
|
|
// arguments in .user file
|
|
cmd.addArgs(aspect<ArgumentsAspect>()->arguments(), CommandLine::Raw);
|
|
|
|
// arguments from .qmlproject file
|
|
const QmlBuildSystem *bs = qobject_cast<QmlBuildSystem *>(target->buildSystem());
|
|
for (const QString &importPath : bs->customImportPaths()) {
|
|
cmd.addArg("-I");
|
|
cmd.addArg(bs->targetDirectory().pathAppended(importPath).path());
|
|
}
|
|
|
|
for (const QString &fileSelector : bs->customFileSelectors()) {
|
|
cmd.addArg("-S");
|
|
cmd.addArg(fileSelector);
|
|
}
|
|
|
|
if (qmlRuntime.osType() == OsTypeWindows && bs->forceFreeType()) {
|
|
cmd.addArg("-platform");
|
|
cmd.addArg("windows:fontengine=freetype");
|
|
}
|
|
|
|
if (bs->qt6Project() && bs->widgetApp()) {
|
|
cmd.addArg("--apptype");
|
|
cmd.addArg("widget");
|
|
}
|
|
|
|
const FilePath main = bs->targetFile(mainScript());
|
|
|
|
if (!main.isEmpty())
|
|
cmd.addArg(main.path());
|
|
|
|
return cmd;
|
|
});
|
|
|
|
m_qmlMainFileAspect = addAspect<QmlMainFileAspect>(target);
|
|
connect(m_qmlMainFileAspect, &QmlMainFileAspect::changed, this, &RunConfiguration::update);
|
|
|
|
createQtVersionAspect();
|
|
|
|
connect(target, &Target::kitChanged, this, &RunConfiguration::update);
|
|
|
|
m_multiLanguageAspect = addAspect<QmlMultiLanguageAspect>(target);
|
|
auto buildSystem = qobject_cast<const QmlBuildSystem *>(activeBuildSystem());
|
|
if (buildSystem)
|
|
m_multiLanguageAspect->setValue(buildSystem->multilanguageSupport());
|
|
|
|
auto envAspect = addAspect<EnvironmentAspect>();
|
|
connect(m_multiLanguageAspect,
|
|
&QmlMultiLanguageAspect::changed,
|
|
envAspect,
|
|
&EnvironmentAspect::environmentChanged);
|
|
|
|
auto envModifier = [this](Environment env) {
|
|
if (auto bs = qobject_cast<const QmlBuildSystem *>(activeBuildSystem()))
|
|
env.modify(bs->environment());
|
|
|
|
if (m_multiLanguageAspect && m_multiLanguageAspect->value()
|
|
&& !m_multiLanguageAspect->databaseFilePath().isEmpty()) {
|
|
env.set("QT_MULTILANGUAGE_DATABASE",
|
|
m_multiLanguageAspect->databaseFilePath().toString());
|
|
env.set("QT_MULTILANGUAGE_LANGUAGE", m_multiLanguageAspect->currentLocale());
|
|
} else {
|
|
env.unset("QT_MULTILANGUAGE_DATABASE");
|
|
env.unset("QT_MULTILANGUAGE_LANGUAGE");
|
|
}
|
|
return env;
|
|
};
|
|
|
|
const Id deviceTypeId = DeviceTypeKitAspect::deviceTypeId(target->kit());
|
|
if (deviceTypeId == ProjectExplorer::Constants::DESKTOP_DEVICE_TYPE) {
|
|
envAspect->addPreferredBaseEnvironment(Tr::tr("System Environment"), [envModifier] {
|
|
return envModifier(Environment::systemEnvironment());
|
|
});
|
|
}
|
|
|
|
envAspect->addSupportedBaseEnvironment(Tr::tr("Clean Environment"), [envModifier] {
|
|
Environment environment;
|
|
return envModifier(environment);
|
|
});
|
|
|
|
if (HostOsInfo::isAnyUnixHost())
|
|
addAspect<X11ForwardingAspect>(macroExpander());
|
|
|
|
setRunnableModifier([this](Runnable &r) {
|
|
const QmlBuildSystem *bs = static_cast<QmlBuildSystem *>(activeBuildSystem());
|
|
r.workingDirectory = bs->targetDirectory();
|
|
if (const auto * const forwardingAspect = aspect<X11ForwardingAspect>())
|
|
r.extraData.insert("Ssh.X11ForwardToDisplay", forwardingAspect->display());
|
|
});
|
|
|
|
setDisplayName(Tr::tr("QML Utility", "QMLRunConfiguration display name."));
|
|
update();
|
|
}
|
|
|
|
QString QmlProjectRunConfiguration::disabledReason() const
|
|
{
|
|
if (mainScript().isEmpty())
|
|
return Tr::tr("No script file to execute.");
|
|
|
|
const FilePath viewer = qmlRuntimeFilePath();
|
|
if (DeviceTypeKitAspect::deviceTypeId(kit())
|
|
== ProjectExplorer::Constants::DESKTOP_DEVICE_TYPE
|
|
&& !viewer.exists()) {
|
|
return Tr::tr("No QML utility found.");
|
|
}
|
|
if (viewer.isEmpty())
|
|
return Tr::tr("No QML utility specified for target device.");
|
|
return RunConfiguration::disabledReason();
|
|
}
|
|
|
|
FilePath QmlProjectRunConfiguration::qmlRuntimeFilePath() const
|
|
{
|
|
usePuppetAsQmlRuntime = false;
|
|
// Give precedence to the manual override in the run configuration.
|
|
const FilePath qmlViewer = m_qmlViewerAspect->filePath();
|
|
if (!qmlViewer.isEmpty())
|
|
return qmlViewer;
|
|
|
|
Kit *kit = target()->kit();
|
|
|
|
// We might not have a full Qt version for building, but the device
|
|
// might know what is good for running.
|
|
if (IDevice::ConstPtr dev = DeviceKitAspect::device(kit)) {
|
|
const FilePath qmlRuntime = dev->qmlRunCommand();
|
|
if (!qmlRuntime.isEmpty())
|
|
return qmlRuntime;
|
|
}
|
|
|
|
// The Qt version might know. That's the "build" Qt version,
|
|
// i.e. not necessarily something the device can use, but the
|
|
// device had its chance above.
|
|
if (QtVersion *version = QtKitAspect::qtVersion(kit)) {
|
|
// look for puppet as qmlruntime only in QtStudio Qt versions
|
|
if (version->features().contains("QtStudio") &&
|
|
version->qtVersion().majorVersion() > 5) {
|
|
|
|
auto [workingDirectoryPath, puppetPath] = QmlDesigner::QmlPuppetPaths::qmlPuppetPaths(
|
|
target(), QmlDesigner::QmlDesignerBasePlugin::settings());
|
|
if (!puppetPath.isEmpty()) {
|
|
usePuppetAsQmlRuntime = true;
|
|
return puppetPath;
|
|
}
|
|
}
|
|
const FilePath qmlRuntime = version->qmlRuntimeFilePath();
|
|
if (!qmlRuntime.isEmpty())
|
|
return qmlRuntime;
|
|
}
|
|
|
|
// If not given explicitly by run device, nor Qt, try to pick
|
|
// it from $PATH on the run device.
|
|
return "qml";
|
|
}
|
|
|
|
void QmlProjectRunConfiguration::createQtVersionAspect()
|
|
{
|
|
if (!QmlProject::isQtDesignStudio())
|
|
return;
|
|
|
|
m_qtversionAspect = addAspect<SelectionAspect>();
|
|
m_qtversionAspect->setDisplayStyle(SelectionAspect::DisplayStyle::ComboBox);
|
|
m_qtversionAspect->setLabelText(Tr::tr("Qt Version:"));
|
|
m_qtversionAspect->setSettingsKey("QmlProjectManager.kit");
|
|
|
|
Kit *kit = target()->kit();
|
|
QtVersion *version = QtKitAspect::qtVersion(kit);
|
|
|
|
if (version) {
|
|
const QmlBuildSystem *buildSystem = qobject_cast<QmlBuildSystem *>(target()->buildSystem());
|
|
const bool isQt6Project = buildSystem && buildSystem->qt6Project();
|
|
|
|
if (isQt6Project) {
|
|
m_qtversionAspect->addOption(Tr::tr("Qt 6"));
|
|
m_qtversionAspect->setReadOnly(true);
|
|
} else { /* Only if this is not a Qt 6 project changing kits makes sense */
|
|
m_qtversionAspect->addOption(Tr::tr("Qt 5"));
|
|
m_qtversionAspect->addOption(Tr::tr("Qt 6"));
|
|
|
|
const int valueForVersion = version->qtVersion().majorVersion() == 6 ? 1 : 0;
|
|
|
|
m_qtversionAspect->setValue(valueForVersion);
|
|
|
|
connect(m_qtversionAspect, &SelectionAspect::changed, this, [&]() {
|
|
QTC_ASSERT(target(), return );
|
|
auto project = target()->project();
|
|
QTC_ASSERT(project, return );
|
|
|
|
int oldValue = !m_qtversionAspect->value();
|
|
const int preferedQtVersion = m_qtversionAspect->value() > 0 ? 6 : 5;
|
|
Kit *currentKit = target()->kit();
|
|
|
|
const QList<Kit *> kits = Utils::filtered(KitManager::kits(), [&](const Kit *k) {
|
|
QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(k);
|
|
return (version && version->qtVersion().majorVersion() == preferedQtVersion)
|
|
&& DeviceTypeKitAspect::deviceTypeId(k)
|
|
== ProjectExplorer::Constants::DESKTOP_DEVICE_TYPE;
|
|
});
|
|
|
|
if (kits.contains(currentKit))
|
|
return;
|
|
|
|
if (!kits.isEmpty()) {
|
|
auto newTarget = target()->project()->target(kits.first());
|
|
if (!newTarget)
|
|
newTarget = project->addTargetForKit(kits.first());
|
|
|
|
project->setActiveTarget(newTarget, SetActive::Cascade);
|
|
|
|
/* Reset the aspect. We changed the target and this aspect should not change. */
|
|
m_qtversionAspect->blockSignals(true);
|
|
m_qtversionAspect->setValue(oldValue);
|
|
m_qtversionAspect->blockSignals(false);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
bool QmlProjectRunConfiguration::isEnabled() const
|
|
{
|
|
return m_qmlMainFileAspect->isQmlFilePresent() && !commandLine().executable().isEmpty()
|
|
&& activeBuildSystem()->hasParsingData();
|
|
}
|
|
|
|
FilePath QmlProjectRunConfiguration::mainScript() const
|
|
{
|
|
return m_qmlMainFileAspect->mainScript();
|
|
}
|
|
|
|
// QmlProjectRunConfigurationFactory
|
|
|
|
QmlProjectRunConfigurationFactory::QmlProjectRunConfigurationFactory()
|
|
: FixedRunConfigurationFactory(Tr::tr("QML Runtime"), false)
|
|
{
|
|
registerRunConfiguration<QmlProjectRunConfiguration>
|
|
("QmlProjectManager.QmlRunConfiguration.Qml");
|
|
addSupportedProjectType(QmlProjectManager::Constants::QML_PROJECT_ID);
|
|
}
|
|
|
|
} // QmlProjectManager::Internal
|