forked from qt-creator/qt-creator
Change-Id: Ide05c2ca859454c1745e5c243af6a3d05131194c Reviewed-by: Jarek Kobus <jaroslaw.kobus@qt.io>
433 lines
16 KiB
C++
433 lines
16 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 "iosrunconfiguration.h"
|
|
|
|
#include "iosconstants.h"
|
|
#include "iosdevice.h"
|
|
#include "iostr.h"
|
|
#include "simulatorcontrol.h"
|
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
#include <projectexplorer/buildstep.h>
|
|
#include <projectexplorer/buildsteplist.h>
|
|
#include <projectexplorer/deployconfiguration.h>
|
|
#include <projectexplorer/devicesupport/devicemanager.h>
|
|
#include <projectexplorer/kitaspects.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/projectnodes.h>
|
|
#include <projectexplorer/runconfigurationaspects.h>
|
|
#include <projectexplorer/target.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/filepath.h>
|
|
#include <utils/layoutbuilder.h>
|
|
#include <utils/process.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QAction>
|
|
#include <QApplication>
|
|
#include <QComboBox>
|
|
#include <QHeaderView>
|
|
#include <QLabel>
|
|
#include <QLineEdit>
|
|
#include <QList>
|
|
#include <QVariant>
|
|
#include <QWidget>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace Utils;
|
|
|
|
namespace Ios::Internal {
|
|
|
|
const char deviceTypeKey[] = "Ios.device_type";
|
|
|
|
static QString displayName(const SimulatorInfo &device)
|
|
{
|
|
return QString("%1, %2").arg(device.name).arg(device.runtimeName);
|
|
}
|
|
|
|
static IosDeviceType toIosDeviceType(const SimulatorInfo &device)
|
|
{
|
|
IosDeviceType iosDeviceType(IosDeviceType::SimulatedDevice,
|
|
device.identifier,
|
|
displayName(device));
|
|
return iosDeviceType;
|
|
}
|
|
|
|
IosRunConfiguration::IosRunConfiguration(Target *target, Id id)
|
|
: RunConfiguration(target, id), iosDeviceType(this, this)
|
|
{
|
|
executable.setDeviceSelector(target, ExecutableAspect::RunDevice);
|
|
|
|
arguments.setMacroExpander(macroExpander());
|
|
|
|
setUpdater([this, target] {
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(target->kit());
|
|
const QString devName = dev ? dev->displayName() : IosDevice::name();
|
|
setDefaultDisplayName(Tr::tr("Run on %1").arg(devName));
|
|
setDisplayName(Tr::tr("Run %1 on %2").arg(applicationName()).arg(devName));
|
|
|
|
executable.setExecutable(localExecutable());
|
|
iosDeviceType.updateDeviceType();
|
|
});
|
|
}
|
|
|
|
void IosDeviceTypeAspect::deviceChanges()
|
|
{
|
|
m_runConfiguration->update();
|
|
}
|
|
|
|
void IosDeviceTypeAspect::updateDeviceType()
|
|
{
|
|
if (DeviceTypeKitAspect::deviceTypeId(m_runConfiguration->kit())
|
|
== Constants::IOS_DEVICE_TYPE)
|
|
m_deviceType = IosDeviceType(IosDeviceType::IosDevice);
|
|
else if (m_deviceType.type == IosDeviceType::IosDevice)
|
|
m_deviceType = IosDeviceType(IosDeviceType::SimulatedDevice);
|
|
}
|
|
|
|
bool IosRunConfiguration::isEnabled(Id runMode) const
|
|
{
|
|
Utils::Id devType = DeviceTypeKitAspect::deviceTypeId(kit());
|
|
if (devType != Constants::IOS_DEVICE_TYPE && devType != Constants::IOS_SIMULATOR_TYPE)
|
|
return false;
|
|
if (devType == Constants::IOS_SIMULATOR_TYPE)
|
|
return true;
|
|
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(kit());
|
|
if (!dev || dev->deviceState() != IDevice::DeviceReadyToUse)
|
|
return false;
|
|
|
|
IosDevice::ConstPtr iosdevice = std::dynamic_pointer_cast<const IosDevice>(dev);
|
|
if (iosdevice && iosdevice->handler() == IosDevice::Handler::DeviceCtl
|
|
&& runMode != ProjectExplorer::Constants::NORMAL_RUN_MODE) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
QString IosRunConfiguration::applicationName() const
|
|
{
|
|
if (ProjectNode *node = project()->findNodeForBuildKey(buildKey()))
|
|
return node->data(Constants::IosTarget).toString();
|
|
|
|
return QString();
|
|
}
|
|
|
|
FilePath IosRunConfiguration::bundleDirectory() const
|
|
{
|
|
Utils::Id devType = DeviceTypeKitAspect::deviceTypeId(kit());
|
|
bool isDevice = (devType == Constants::IOS_DEVICE_TYPE);
|
|
if (!isDevice && devType != Constants::IOS_SIMULATOR_TYPE) {
|
|
qCWarning(iosLog) << "unexpected device type in bundleDirForTarget: " << devType.toString();
|
|
return {};
|
|
}
|
|
FilePath res;
|
|
bool shouldAppendBuildTypeAndPlatform = true;
|
|
if (BuildConfiguration *bc = target()->activeBuildConfiguration()) {
|
|
Project *project = target()->project();
|
|
if (ProjectNode *node = project->findNodeForBuildKey(buildKey())) {
|
|
QString pathStr = node->data(Constants::IosBuildDir).toString();
|
|
const QString cmakeGenerator = node->data(Constants::IosCmakeGenerator).toString();
|
|
|
|
if (cmakeGenerator.isEmpty()) {
|
|
// qmake node gives absolute IosBuildDir
|
|
res = FilePath::fromString(pathStr);
|
|
} else {
|
|
// CMake node gives IosBuildDir relative to root build directory
|
|
|
|
bool useCmakePath = true;
|
|
|
|
if (pathStr.isEmpty())
|
|
useCmakePath = false;
|
|
|
|
if (useCmakePath && cmakeGenerator == "Xcode") {
|
|
// When generating Xcode project, CMake may put a "${EFFECTIVE_PLATFORM_NAME}" macro,
|
|
// which is expanded by Xcode at build time.
|
|
// To get an actual executable path at configure time, replace this macro here
|
|
// depending on the device type.
|
|
|
|
const QString before = "${EFFECTIVE_PLATFORM_NAME}";
|
|
|
|
int idx = pathStr.indexOf(before);
|
|
|
|
if (idx == -1) {
|
|
useCmakePath = false;
|
|
} else {
|
|
QString after;
|
|
if (isDevice)
|
|
after = "-iphoneos";
|
|
else
|
|
after = "-iphonesimulator";
|
|
|
|
pathStr.replace(idx, before.length(), after);
|
|
}
|
|
}
|
|
|
|
if (useCmakePath) {
|
|
// With Ninja generator IosBuildDir may be just "." when executable is in the root directory,
|
|
// so use canonical path to ensure that redundand dot is removed.
|
|
res = bc->buildDirectory().pathAppended(pathStr).canonicalPath();
|
|
// All done with path provided by CMake
|
|
shouldAppendBuildTypeAndPlatform = false;
|
|
} else {
|
|
res = bc->buildDirectory();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (res.isEmpty()) {
|
|
// Fallback
|
|
res = bc->buildDirectory();
|
|
shouldAppendBuildTypeAndPlatform = true;
|
|
}
|
|
|
|
if (shouldAppendBuildTypeAndPlatform) {
|
|
switch (bc->buildType()) {
|
|
case BuildConfiguration::Debug :
|
|
case BuildConfiguration::Unknown :
|
|
if (isDevice)
|
|
res = res / "Debug-iphoneos";
|
|
else
|
|
res = res.pathAppended("Debug-iphonesimulator");
|
|
break;
|
|
case BuildConfiguration::Profile :
|
|
case BuildConfiguration::Release :
|
|
if (isDevice)
|
|
res = res.pathAppended("Release-iphoneos");
|
|
else
|
|
res = res.pathAppended("Release-iphonesimulator");
|
|
break;
|
|
default:
|
|
qCWarning(iosLog) << "IosBuildStep had an unknown buildType "
|
|
<< target()->activeBuildConfiguration()->buildType();
|
|
}
|
|
}
|
|
}
|
|
return res.pathAppended(applicationName() + ".app");
|
|
}
|
|
|
|
FilePath IosRunConfiguration::localExecutable() const
|
|
{
|
|
return bundleDirectory().pathAppended(applicationName());
|
|
}
|
|
|
|
void IosDeviceTypeAspect::fromMap(const Store &map)
|
|
{
|
|
bool deviceTypeIsInt;
|
|
map.value(deviceTypeKey).toInt(&deviceTypeIsInt);
|
|
if (deviceTypeIsInt || !m_deviceType.fromMap(storeFromVariant(map.value(deviceTypeKey))))
|
|
updateDeviceType();
|
|
|
|
m_runConfiguration->update();
|
|
}
|
|
|
|
void IosDeviceTypeAspect::toMap(Store &map) const
|
|
{
|
|
map.insert(deviceTypeKey, QVariant::fromValue(deviceType().toMap()));
|
|
}
|
|
|
|
QString IosRunConfiguration::disabledReason(Id runMode) const
|
|
{
|
|
Utils::Id devType = DeviceTypeKitAspect::deviceTypeId(kit());
|
|
if (devType != Constants::IOS_DEVICE_TYPE && devType != Constants::IOS_SIMULATOR_TYPE)
|
|
return Tr::tr("Kit has incorrect device type for running on iOS devices.");
|
|
IDevice::ConstPtr dev = DeviceKitAspect::device(kit());
|
|
QString validDevName;
|
|
bool hasConncetedDev = false;
|
|
if (devType == Constants::IOS_DEVICE_TYPE) {
|
|
DeviceManager *dm = DeviceManager::instance();
|
|
for (int idev = 0; idev < dm->deviceCount(); ++idev) {
|
|
IDevice::ConstPtr availDev = dm->deviceAt(idev);
|
|
if (availDev && availDev->type() == Constants::IOS_DEVICE_TYPE) {
|
|
if (availDev->deviceState() == IDevice::DeviceReadyToUse) {
|
|
validDevName += QLatin1Char(' ');
|
|
validDevName += availDev->displayName();
|
|
} else if (availDev->deviceState() == IDevice::DeviceConnected) {
|
|
hasConncetedDev = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!dev) {
|
|
if (!validDevName.isEmpty())
|
|
return Tr::tr("No device chosen. Select %1.").arg(validDevName); // should not happen
|
|
else if (hasConncetedDev)
|
|
return Tr::tr("No device chosen. Enable developer mode on a device."); // should not happen
|
|
else
|
|
return Tr::tr("No device available.");
|
|
} else if (devType == Constants::IOS_DEVICE_TYPE) {
|
|
switch (dev->deviceState()) {
|
|
case IDevice::DeviceReadyToUse:
|
|
break;
|
|
case IDevice::DeviceConnected:
|
|
return Tr::tr("To use this device you need to enable developer mode on it.");
|
|
case IDevice::DeviceDisconnected:
|
|
case IDevice::DeviceStateUnknown:
|
|
if (!validDevName.isEmpty())
|
|
return Tr::tr("%1 is not connected. Select %2?")
|
|
.arg(dev->displayName(), validDevName);
|
|
else if (hasConncetedDev)
|
|
return Tr::tr("%1 is not connected. Enable developer mode on a device?")
|
|
.arg(dev->displayName());
|
|
else
|
|
return Tr::tr("%1 is not connected.").arg(dev->displayName());
|
|
}
|
|
IosDevice::ConstPtr iosdevice = std::dynamic_pointer_cast<const IosDevice>(dev);
|
|
if (iosdevice && iosdevice->handler() == IosDevice::Handler::DeviceCtl
|
|
&& runMode != ProjectExplorer::Constants::NORMAL_RUN_MODE) {
|
|
return Tr::tr("Debugging and profiling is currently not supported for devices with iOS "
|
|
"17 and later.");
|
|
}
|
|
}
|
|
return RunConfiguration::disabledReason(runMode);
|
|
}
|
|
|
|
IosDeviceType IosRunConfiguration::deviceType() const
|
|
{
|
|
return iosDeviceType.deviceType();
|
|
}
|
|
|
|
IosDeviceType IosDeviceTypeAspect::deviceType() const
|
|
{
|
|
if (m_deviceType.type == IosDeviceType::SimulatedDevice) {
|
|
QList<SimulatorInfo> availableSimulators = SimulatorControl::availableSimulators();
|
|
if (availableSimulators.isEmpty())
|
|
return m_deviceType;
|
|
if (Utils::contains(availableSimulators,
|
|
Utils::equal(&SimulatorInfo::identifier, m_deviceType.identifier))) {
|
|
return m_deviceType;
|
|
}
|
|
const QStringList parts = m_deviceType.displayName.split(QLatin1Char(','));
|
|
if (parts.count() < 2)
|
|
return toIosDeviceType(availableSimulators.last());
|
|
|
|
QList<SimulatorInfo> eligibleDevices;
|
|
eligibleDevices = Utils::filtered(availableSimulators, [parts](const SimulatorInfo &info) {
|
|
return info.name == parts.at(0) && info.runtimeName == parts.at(1);
|
|
});
|
|
return toIosDeviceType(eligibleDevices.isEmpty() ? availableSimulators.last()
|
|
: eligibleDevices.last());
|
|
}
|
|
return m_deviceType;
|
|
}
|
|
|
|
void IosDeviceTypeAspect::setDeviceType(const IosDeviceType &deviceType)
|
|
{
|
|
m_deviceType = deviceType;
|
|
}
|
|
|
|
IosDeviceTypeAspect::IosDeviceTypeAspect(AspectContainer *container, IosRunConfiguration *rc)
|
|
: BaseAspect(container), m_runConfiguration(rc)
|
|
{
|
|
addDataExtractor(this, &IosDeviceTypeAspect::deviceType, &Data::deviceType);
|
|
addDataExtractor(this, &IosDeviceTypeAspect::bundleDirectory, &Data::bundleDirectory);
|
|
addDataExtractor(this, &IosDeviceTypeAspect::applicationName, &Data::applicationName);
|
|
addDataExtractor(this, &IosDeviceTypeAspect::localExecutable, &Data::localExecutable);
|
|
|
|
connect(DeviceManager::instance(), &DeviceManager::updated,
|
|
this, &IosDeviceTypeAspect::deviceChanges);
|
|
connect(KitManager::instance(), &KitManager::kitsChanged,
|
|
this, &IosDeviceTypeAspect::deviceChanges);
|
|
}
|
|
|
|
void IosDeviceTypeAspect::addToLayout(Layouting::LayoutItem &parent)
|
|
{
|
|
m_deviceTypeComboBox = new QComboBox;
|
|
m_deviceTypeComboBox->setModel(&m_deviceTypeModel);
|
|
|
|
m_deviceTypeLabel = new QLabel(Tr::tr("Device type:"));
|
|
|
|
parent.addItems({m_deviceTypeLabel, m_deviceTypeComboBox});
|
|
|
|
updateValues();
|
|
|
|
connect(m_deviceTypeComboBox, &QComboBox::currentIndexChanged,
|
|
this, &IosDeviceTypeAspect::setDeviceTypeIndex);
|
|
}
|
|
|
|
void IosDeviceTypeAspect::setDeviceTypeIndex(int devIndex)
|
|
{
|
|
QVariant selectedDev = m_deviceTypeModel.data(m_deviceTypeModel.index(devIndex, 0), Qt::UserRole + 1);
|
|
if (selectedDev.isValid())
|
|
setDeviceType(toIosDeviceType(selectedDev.value<SimulatorInfo>()));
|
|
}
|
|
|
|
|
|
void IosDeviceTypeAspect::updateValues()
|
|
{
|
|
bool showDeviceSelector = deviceType().type != IosDeviceType::IosDevice;
|
|
m_deviceTypeLabel->setVisible(showDeviceSelector);
|
|
m_deviceTypeComboBox->setVisible(showDeviceSelector);
|
|
if (showDeviceSelector && m_deviceTypeModel.rowCount() == 0) {
|
|
const QList<SimulatorInfo> devices = SimulatorControl::availableSimulators();
|
|
for (const SimulatorInfo &device : devices) {
|
|
QStandardItem *item = new QStandardItem(Internal::displayName(device));
|
|
QVariant v;
|
|
v.setValue(device);
|
|
item->setData(v);
|
|
m_deviceTypeModel.appendRow(item);
|
|
}
|
|
}
|
|
|
|
IosDeviceType currentDType = deviceType();
|
|
QVariant currentData = m_deviceTypeComboBox->currentData();
|
|
if (currentDType.type == IosDeviceType::SimulatedDevice && !currentDType.identifier.isEmpty()
|
|
&& (!currentData.isValid()
|
|
|| currentDType != toIosDeviceType(currentData.value<SimulatorInfo>())))
|
|
{
|
|
bool didSet = false;
|
|
for (int i = 0; m_deviceTypeModel.hasIndex(i, 0); ++i) {
|
|
QVariant vData = m_deviceTypeModel.data(m_deviceTypeModel.index(i, 0), Qt::UserRole + 1);
|
|
SimulatorInfo dType = vData.value<SimulatorInfo>();
|
|
if (dType.identifier == currentDType.identifier) {
|
|
m_deviceTypeComboBox->setCurrentIndex(i);
|
|
didSet = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!didSet) {
|
|
qCWarning(iosLog) << "could not set " << currentDType << " as it is not in model";
|
|
}
|
|
}
|
|
}
|
|
|
|
FilePath IosDeviceTypeAspect::bundleDirectory() const
|
|
{
|
|
return m_runConfiguration->bundleDirectory();
|
|
}
|
|
|
|
QString IosDeviceTypeAspect::applicationName() const
|
|
{
|
|
return m_runConfiguration->applicationName();
|
|
}
|
|
|
|
FilePath IosDeviceTypeAspect::localExecutable() const
|
|
{
|
|
return m_runConfiguration->localExecutable();
|
|
}
|
|
|
|
// IosRunConfigurationFactory
|
|
|
|
class IosRunConfigurationFactory final : public RunConfigurationFactory
|
|
{
|
|
public:
|
|
IosRunConfigurationFactory()
|
|
{
|
|
registerRunConfiguration<IosRunConfiguration>(Constants::IOS_RUNCONFIG_ID);
|
|
addSupportedTargetDeviceType(Constants::IOS_DEVICE_TYPE);
|
|
addSupportedTargetDeviceType(Constants::IOS_SIMULATOR_TYPE);
|
|
}
|
|
};
|
|
|
|
void setupIosRunConfiguration()
|
|
{
|
|
static IosRunConfigurationFactory theIosRunConfigurationFactory;
|
|
}
|
|
|
|
} // Ios::Internal
|