forked from qt-creator/qt-creator
Change-Id: I100b16a1fd74c4528f61f681bbeb5cdb4adddee5 Reviewed-by: <github-actions-qt-creator@cristianadam.eu> Reviewed-by: Alessandro Portale <alessandro.portale@qt.io> Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
1134 lines
43 KiB
C++
1134 lines
43 KiB
C++
// Copyright (C) 2016 BogDan Vatra <bog_dan_ro@yahoo.com>
|
|
// 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 "androidbuildapkstep.h"
|
|
#include "androidconfigurations.h"
|
|
#include "androidconstants.h"
|
|
#include "androidcreatekeystorecertificate.h"
|
|
#include "androidextralibrarylistmodel.h"
|
|
#include "androidmanager.h"
|
|
#include "androidqtversion.h"
|
|
#include "androidsdkmanager.h"
|
|
#include "androidtr.h"
|
|
#include "certificatesmodel.h"
|
|
#include "createandroidmanifestwizard.h"
|
|
#include "javaparser.h"
|
|
|
|
#include <coreplugin/fileutils.h>
|
|
#include <coreplugin/icore.h>
|
|
|
|
#include <projectexplorer/buildconfiguration.h>
|
|
#include <projectexplorer/buildstep.h>
|
|
#include <projectexplorer/buildsteplist.h>
|
|
#include <projectexplorer/buildsystem.h>
|
|
#include <projectexplorer/processparameters.h>
|
|
#include <projectexplorer/projectexplorerconstants.h>
|
|
#include <projectexplorer/project.h>
|
|
#include <projectexplorer/projectnodes.h>
|
|
#include <projectexplorer/target.h>
|
|
#include <projectexplorer/taskhub.h>
|
|
|
|
#include <qtsupport/qtkitinformation.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/fancylineedit.h>
|
|
#include <utils/infolabel.h>
|
|
#include <utils/layoutbuilder.h>
|
|
#include <utils/pathchooser.h>
|
|
#include <utils/process.h>
|
|
|
|
#include <QCheckBox>
|
|
#include <QComboBox>
|
|
#include <QDateTime>
|
|
#include <QDialogButtonBox>
|
|
#include <QFileDialog>
|
|
#include <QFormLayout>
|
|
#include <QGroupBox>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QLabel>
|
|
#include <QLineEdit>
|
|
#include <QListView>
|
|
#include <QLoggingCategory>
|
|
#include <QMessageBox>
|
|
#include <QProcess>
|
|
#include <QPushButton>
|
|
#include <QTimer>
|
|
#include <QToolButton>
|
|
|
|
#include <algorithm>
|
|
#include <memory>
|
|
|
|
using namespace ProjectExplorer;
|
|
using namespace QtSupport;
|
|
using namespace Utils;
|
|
|
|
namespace Android {
|
|
namespace Internal {
|
|
|
|
static Q_LOGGING_CATEGORY(buildapkstepLog, "qtc.android.build.androidbuildapkstep", QtWarningMsg)
|
|
|
|
const char KeystoreLocationKey[] = "KeystoreLocation";
|
|
const char BuildTargetSdkKey[] = "BuildTargetSdk";
|
|
const char BuildToolsVersionKey[] = "BuildToolsVersion";
|
|
const char VerboseOutputKey[] = "VerboseOutput";
|
|
|
|
class PasswordInputDialog : public QDialog
|
|
{
|
|
public:
|
|
enum Context{
|
|
KeystorePassword = 1,
|
|
CertificatePassword
|
|
};
|
|
|
|
PasswordInputDialog(Context context, std::function<bool (const QString &)> callback,
|
|
const QString &extraContextStr, QWidget *parent = nullptr);
|
|
|
|
static QString getPassword(Context context, std::function<bool (const QString &)> callback,
|
|
const QString &extraContextStr, bool *ok = nullptr,
|
|
QWidget *parent = nullptr);
|
|
|
|
private:
|
|
std::function<bool (const QString &)> verifyCallback = [](const QString &) { return true; };
|
|
QLabel *inputContextlabel = new QLabel(this);
|
|
QLineEdit *inputEdit = new QLineEdit(this);
|
|
Utils::InfoLabel *warningLabel = new Utils::InfoLabel(::Android::Tr::tr("Incorrect password."),
|
|
Utils::InfoLabel::Warning, this);
|
|
QDialogButtonBox *buttonBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel,
|
|
this);
|
|
};
|
|
|
|
// AndroidBuildApkWidget
|
|
|
|
class AndroidBuildApkWidget : public QWidget
|
|
{
|
|
public:
|
|
explicit AndroidBuildApkWidget(AndroidBuildApkStep *step);
|
|
|
|
private:
|
|
void setCertificates();
|
|
void updateSigningWarning();
|
|
void signPackageCheckBoxToggled(bool checked);
|
|
void onOpenSslCheckBoxChanged();
|
|
bool isOpenSslLibsIncluded();
|
|
FilePath appProjectFilePath() const;
|
|
QString openSslIncludeFileContent(const FilePath &projectPath);
|
|
|
|
QWidget *createApplicationGroup();
|
|
QWidget *createSignPackageGroup();
|
|
QWidget *createAdvancedGroup();
|
|
QWidget *createAdditionalLibrariesGroup();
|
|
|
|
private:
|
|
AndroidBuildApkStep *m_step = nullptr;
|
|
QCheckBox *m_signPackageCheckBox = nullptr;
|
|
InfoLabel *m_signingDebugWarningLabel = nullptr;
|
|
QComboBox *m_certificatesAliasComboBox = nullptr;
|
|
QCheckBox *m_addDebuggerCheckBox = nullptr;
|
|
QCheckBox *m_openSslCheckBox = nullptr;
|
|
};
|
|
|
|
AndroidBuildApkWidget::AndroidBuildApkWidget(AndroidBuildApkStep *step)
|
|
: m_step(step)
|
|
{
|
|
using namespace Layouting;
|
|
Column {
|
|
createSignPackageGroup(),
|
|
createApplicationGroup(),
|
|
createAdvancedGroup(),
|
|
createAdditionalLibrariesGroup(),
|
|
noMargin
|
|
}.attachTo(this);
|
|
|
|
connect(m_step->buildConfiguration(), &BuildConfiguration::buildTypeChanged,
|
|
this, &AndroidBuildApkWidget::updateSigningWarning);
|
|
|
|
connect(m_signPackageCheckBox, &QAbstractButton::clicked,
|
|
m_addDebuggerCheckBox, &QWidget::setEnabled);
|
|
|
|
signPackageCheckBoxToggled(m_step->signPackage());
|
|
updateSigningWarning();
|
|
}
|
|
|
|
QWidget *AndroidBuildApkWidget::createApplicationGroup()
|
|
{
|
|
QtSupport::QtVersion *qt = QtSupport::QtKitAspect::qtVersion(m_step->target()->kit());
|
|
const int minApiSupported = AndroidManager::defaultMinimumSDK(qt);
|
|
QStringList targets = AndroidConfig::apiLevelNamesFor(AndroidConfigurations::sdkManager()->
|
|
filteredSdkPlatforms(minApiSupported));
|
|
targets.removeDuplicates();
|
|
|
|
auto group = new QGroupBox(Tr::tr("Application"), this);
|
|
|
|
auto targetSDKComboBox = new QComboBox();
|
|
targetSDKComboBox->addItems(targets);
|
|
targetSDKComboBox->setCurrentIndex(targets.indexOf(m_step->buildTargetSdk()));
|
|
|
|
connect(targetSDKComboBox, &QComboBox::activated, this, [this, targetSDKComboBox](int idx) {
|
|
const QString sdk = targetSDKComboBox->itemText(idx);
|
|
m_step->setBuildTargetSdk(sdk);
|
|
});
|
|
targetSDKComboBox->setCurrentIndex(targets.indexOf(m_step->buildTargetSdk()));
|
|
|
|
const QList<QVersionNumber> buildToolsVersions = Utils::transform(
|
|
AndroidConfigurations::sdkManager()->filteredBuildTools(minApiSupported),
|
|
[](const BuildTools *pkg) {
|
|
return pkg->revision();
|
|
});
|
|
|
|
auto buildToolsSdkComboBox = new QComboBox();
|
|
for (const QVersionNumber &version : buildToolsVersions)
|
|
buildToolsSdkComboBox->addItem(version.toString(), QVariant::fromValue(version));
|
|
connect(buildToolsSdkComboBox, &QComboBox::activated, this,
|
|
[this, buildToolsSdkComboBox](int idx) {
|
|
m_step->setBuildToolsVersion(buildToolsSdkComboBox->itemData(idx).value<QVersionNumber>());
|
|
});
|
|
|
|
const int initIdx = (m_step->buildToolsVersion().majorVersion() < 1)
|
|
? buildToolsVersions.indexOf(buildToolsVersions.last())
|
|
: buildToolsVersions.indexOf(m_step->buildToolsVersion());
|
|
buildToolsSdkComboBox->setCurrentIndex(initIdx);
|
|
|
|
auto formLayout = new QFormLayout(group);
|
|
formLayout->addRow(Tr::tr("Android build-tools version:"), buildToolsSdkComboBox);
|
|
formLayout->addRow(Tr::tr("Android build platform SDK:"), targetSDKComboBox);
|
|
|
|
auto createAndroidTemplatesButton = new QPushButton(Tr::tr("Create Templates"));
|
|
createAndroidTemplatesButton->setToolTip(
|
|
Tr::tr("Create an Android package for Custom Java code, assets, and Gradle configurations."));
|
|
connect(createAndroidTemplatesButton, &QAbstractButton::clicked, this, [this] {
|
|
CreateAndroidManifestWizard wizard(m_step->buildSystem());
|
|
wizard.exec();
|
|
});
|
|
|
|
formLayout->addRow(Tr::tr("Android customization:"), createAndroidTemplatesButton);
|
|
|
|
return group;
|
|
}
|
|
|
|
QWidget *AndroidBuildApkWidget::createSignPackageGroup()
|
|
{
|
|
QSizePolicy sizePolicy(QSizePolicy::Fixed, QSizePolicy::Preferred);
|
|
sizePolicy.setHorizontalStretch(0);
|
|
sizePolicy.setVerticalStretch(0);
|
|
|
|
auto group = new QGroupBox(Tr::tr("Application Signature"), this);
|
|
|
|
auto keystoreLocationLabel = new QLabel(Tr::tr("Keystore:"), group);
|
|
keystoreLocationLabel->setAlignment(Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter);
|
|
|
|
auto keystoreLocationChooser = new PathChooser(group);
|
|
keystoreLocationChooser->setExpectedKind(PathChooser::File);
|
|
keystoreLocationChooser->lineEdit()->setReadOnly(true);
|
|
keystoreLocationChooser->setFilePath(m_step->keystorePath());
|
|
keystoreLocationChooser->setInitialBrowsePathBackup(FileUtils::homePath());
|
|
keystoreLocationChooser->setPromptDialogFilter(Tr::tr("Keystore files (*.keystore *.jks)"));
|
|
keystoreLocationChooser->setPromptDialogTitle(Tr::tr("Select Keystore File"));
|
|
connect(keystoreLocationChooser, &PathChooser::textChanged, this, [this, keystoreLocationChooser] {
|
|
const FilePath file = keystoreLocationChooser->rawFilePath();
|
|
m_step->setKeystorePath(file);
|
|
m_signPackageCheckBox->setChecked(!file.isEmpty());
|
|
if (!file.isEmpty())
|
|
setCertificates();
|
|
});
|
|
|
|
auto keystoreCreateButton = new QPushButton(Tr::tr("Create..."), group);
|
|
connect(keystoreCreateButton, &QAbstractButton::clicked, this, [this, keystoreLocationChooser] {
|
|
AndroidCreateKeystoreCertificate d;
|
|
if (d.exec() != QDialog::Accepted)
|
|
return;
|
|
keystoreLocationChooser->setFilePath(d.keystoreFilePath());
|
|
m_step->setKeystorePath(d.keystoreFilePath());
|
|
m_step->setKeystorePassword(d.keystorePassword());
|
|
m_step->setCertificateAlias(d.certificateAlias());
|
|
m_step->setCertificatePassword(d.certificatePassword());
|
|
setCertificates();
|
|
});
|
|
|
|
m_signPackageCheckBox = new QCheckBox(Tr::tr("Sign package"), group);
|
|
m_signPackageCheckBox->setChecked(m_step->signPackage());
|
|
|
|
m_signingDebugWarningLabel = new Utils::InfoLabel(Tr::tr("Signing a debug package"),
|
|
Utils::InfoLabel::Warning, group);
|
|
m_signingDebugWarningLabel->hide();
|
|
|
|
auto certificateAliasLabel = new QLabel(Tr::tr("Certificate alias:"), group);
|
|
certificateAliasLabel->setAlignment(Qt::AlignRight|Qt::AlignVCenter);
|
|
|
|
m_certificatesAliasComboBox = new QComboBox(group);
|
|
m_certificatesAliasComboBox->setEnabled(false);
|
|
m_certificatesAliasComboBox->setSizeAdjustPolicy(QComboBox::AdjustToContents);
|
|
|
|
using namespace Layouting;
|
|
Column {
|
|
Row { keystoreLocationLabel, keystoreLocationChooser, keystoreCreateButton },
|
|
m_signPackageCheckBox,
|
|
Row { m_signingDebugWarningLabel, certificateAliasLabel, m_certificatesAliasComboBox }
|
|
}.attachTo(group);
|
|
|
|
connect(m_signPackageCheckBox, &QAbstractButton::toggled,
|
|
this, &AndroidBuildApkWidget::signPackageCheckBoxToggled);
|
|
|
|
auto updateAlias = [this](int idx) {
|
|
QString alias = m_certificatesAliasComboBox->itemText(idx);
|
|
if (!alias.isEmpty())
|
|
m_step->setCertificateAlias(alias);
|
|
};
|
|
|
|
connect(m_certificatesAliasComboBox, &QComboBox::activated, this, updateAlias);
|
|
connect(m_certificatesAliasComboBox, &QComboBox::currentIndexChanged, this, updateAlias);
|
|
|
|
return group;
|
|
}
|
|
|
|
QWidget *AndroidBuildApkWidget::createAdvancedGroup()
|
|
{
|
|
auto group = new QGroupBox(Tr::tr("Advanced Actions"), this);
|
|
|
|
auto openPackageLocationCheckBox = new QCheckBox(Tr::tr("Open package location after build"), group);
|
|
openPackageLocationCheckBox->setChecked(m_step->openPackageLocation());
|
|
connect(openPackageLocationCheckBox, &QAbstractButton::toggled,
|
|
this, [this](bool checked) { m_step->setOpenPackageLocation(checked); });
|
|
|
|
m_addDebuggerCheckBox = new QCheckBox(Tr::tr("Add debug server"), group);
|
|
m_addDebuggerCheckBox->setEnabled(false);
|
|
m_addDebuggerCheckBox->setToolTip(Tr::tr("Packages debug server with "
|
|
"the APK to enable debugging. For the signed APK this option is unchecked by default."));
|
|
m_addDebuggerCheckBox->setChecked(m_step->addDebugger());
|
|
connect(m_addDebuggerCheckBox, &QAbstractButton::toggled,
|
|
m_step, &AndroidBuildApkStep::setAddDebugger);
|
|
|
|
auto verboseOutputCheckBox = new QCheckBox(Tr::tr("Verbose output"), group);
|
|
verboseOutputCheckBox->setChecked(m_step->verboseOutput());
|
|
|
|
auto vbox = new QVBoxLayout(group);
|
|
QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(m_step->kit());
|
|
if (version && version->qtVersion() >= QVersionNumber(5, 14)) {
|
|
auto buildAAB = new QCheckBox(Tr::tr("Build Android App Bundle (*.aab)"), group);
|
|
buildAAB->setChecked(m_step->buildAAB());
|
|
connect(buildAAB, &QAbstractButton::toggled, m_step, &AndroidBuildApkStep::setBuildAAB);
|
|
vbox->addWidget(buildAAB);
|
|
}
|
|
vbox->addWidget(openPackageLocationCheckBox);
|
|
vbox->addWidget(verboseOutputCheckBox);
|
|
vbox->addWidget(m_addDebuggerCheckBox);
|
|
|
|
connect(verboseOutputCheckBox, &QAbstractButton::toggled,
|
|
this, [this](bool checked) { m_step->setVerboseOutput(checked); });
|
|
|
|
return group;
|
|
}
|
|
|
|
QWidget *AndroidBuildApkWidget::createAdditionalLibrariesGroup()
|
|
{
|
|
auto group = new QGroupBox(Tr::tr("Additional Libraries"));
|
|
group->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
|
|
|
|
auto libsModel = new AndroidExtraLibraryListModel(m_step->buildSystem(), this);
|
|
connect(libsModel, &AndroidExtraLibraryListModel::enabledChanged, this,
|
|
[this, group](const bool enabled) {
|
|
group->setEnabled(enabled);
|
|
m_openSslCheckBox->setChecked(isOpenSslLibsIncluded());
|
|
});
|
|
|
|
auto libsView = new QListView;
|
|
libsView->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
libsView->setToolTip(Tr::tr("List of extra libraries to include in Android package and load on startup."));
|
|
libsView->setModel(libsModel);
|
|
|
|
auto addLibButton = new QPushButton;
|
|
addLibButton->setText(Tr::tr("Add..."));
|
|
addLibButton->setToolTip(Tr::tr("Select library to include in package."));
|
|
addLibButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
|
|
connect(addLibButton, &QAbstractButton::clicked, this, [this, libsModel] {
|
|
QStringList fileNames = QFileDialog::getOpenFileNames(this,
|
|
Tr::tr("Select additional libraries"),
|
|
QDir::homePath(),
|
|
Tr::tr("Libraries (*.so)"));
|
|
if (!fileNames.isEmpty())
|
|
libsModel->addEntries(fileNames);
|
|
});
|
|
|
|
auto removeLibButton = new QPushButton;
|
|
removeLibButton->setText(Tr::tr("Remove"));
|
|
removeLibButton->setToolTip(Tr::tr("Remove currently selected library from list."));
|
|
connect(removeLibButton, &QAbstractButton::clicked, this, [libsModel, libsView] {
|
|
QModelIndexList removeList = libsView->selectionModel()->selectedIndexes();
|
|
libsModel->removeEntries(removeList);
|
|
});
|
|
|
|
m_openSslCheckBox = new QCheckBox(Tr::tr("Include prebuilt OpenSSL libraries"));
|
|
m_openSslCheckBox->setToolTip(Tr::tr("This is useful for apps that use SSL operations. The path "
|
|
"can be defined in Edit > Preferences > Devices > Android."));
|
|
connect(m_openSslCheckBox, &QAbstractButton::clicked, this,
|
|
&AndroidBuildApkWidget::onOpenSslCheckBoxChanged);
|
|
|
|
using namespace Layouting;
|
|
Grid {
|
|
m_openSslCheckBox, br,
|
|
libsView, Column { addLibButton, removeLibButton, st }
|
|
}.attachTo(group);
|
|
|
|
QItemSelectionModel *libSelection = libsView->selectionModel();
|
|
connect(libSelection, &QItemSelectionModel::selectionChanged, this, [libSelection, removeLibButton] {
|
|
removeLibButton->setEnabled(libSelection->hasSelection());
|
|
});
|
|
|
|
Target *target = m_step->target();
|
|
const QString buildKey = target->activeBuildKey();
|
|
const ProjectNode *node = target->project()->findNodeForBuildKey(buildKey);
|
|
group->setEnabled(node && !node->parseInProgress());
|
|
|
|
return group;
|
|
}
|
|
|
|
void AndroidBuildApkWidget::signPackageCheckBoxToggled(bool checked)
|
|
{
|
|
m_certificatesAliasComboBox->setEnabled(checked);
|
|
m_step->setSignPackage(checked);
|
|
m_addDebuggerCheckBox->setChecked(!checked);
|
|
updateSigningWarning();
|
|
if (!checked)
|
|
return;
|
|
if (!m_step->keystorePath().isEmpty())
|
|
setCertificates();
|
|
}
|
|
|
|
void AndroidBuildApkWidget::onOpenSslCheckBoxChanged()
|
|
{
|
|
Utils::FilePath projectPath = appProjectFilePath();
|
|
QFile projectFile(projectPath.toFSPathString());
|
|
if (!projectFile.open(QIODevice::ReadWrite | QIODevice::Text)) {
|
|
qWarning() << "Cannot open project file to add OpenSSL extra libs: " << projectPath;
|
|
return;
|
|
}
|
|
|
|
const QString searchStr = openSslIncludeFileContent(projectPath);
|
|
QTextStream textStream(&projectFile);
|
|
|
|
QString fileContent = textStream.readAll();
|
|
if (!m_openSslCheckBox->isChecked()) {
|
|
fileContent.remove("\n" + searchStr);
|
|
} else if (!fileContent.contains(searchStr, Qt::CaseSensitive)) {
|
|
fileContent.append(searchStr + "\n");
|
|
}
|
|
|
|
projectFile.resize(0);
|
|
textStream << fileContent;
|
|
projectFile.close();
|
|
}
|
|
|
|
FilePath AndroidBuildApkWidget::appProjectFilePath() const
|
|
{
|
|
const FilePath topLevelFile = m_step->buildConfiguration()->buildSystem()->projectFilePath();
|
|
if (topLevelFile.fileName() == "CMakeLists.txt")
|
|
return topLevelFile;
|
|
static const auto isApp = [](Node *n) { return n->asProjectNode()
|
|
&& n->asProjectNode()->productType() == ProductType::App; };
|
|
Node * const appNode = m_step->buildConfiguration()->project()->rootProjectNode()
|
|
->findNode(isApp);
|
|
return appNode ? appNode ->filePath() : topLevelFile;
|
|
}
|
|
|
|
bool AndroidBuildApkWidget::isOpenSslLibsIncluded()
|
|
{
|
|
Utils::FilePath projectPath = appProjectFilePath();
|
|
const QString searchStr = openSslIncludeFileContent(projectPath);
|
|
QFile projectFile(projectPath.toFSPathString());
|
|
projectFile.open(QIODevice::ReadOnly);
|
|
QTextStream textStream(&projectFile);
|
|
QString fileContent = textStream.readAll();
|
|
projectFile.close();
|
|
return fileContent.contains(searchStr, Qt::CaseSensitive);
|
|
}
|
|
|
|
QString AndroidBuildApkWidget::openSslIncludeFileContent(const FilePath &projectPath)
|
|
{
|
|
QString openSslPath = AndroidConfigurations::currentConfig().openSslLocation().toString();
|
|
if (projectPath.endsWith(".pro"))
|
|
return "android: include(" + openSslPath + "/openssl.pri)";
|
|
if (projectPath.endsWith("CMakeLists.txt"))
|
|
return "if (ANDROID)\n include(" + openSslPath + "/CMakeLists.txt)\nendif()";
|
|
return {};
|
|
}
|
|
|
|
void AndroidBuildApkWidget::setCertificates()
|
|
{
|
|
QAbstractItemModel *certificates = m_step->keystoreCertificates();
|
|
if (certificates) {
|
|
m_signPackageCheckBox->setChecked(certificates);
|
|
m_certificatesAliasComboBox->setModel(certificates);
|
|
}
|
|
}
|
|
|
|
void AndroidBuildApkWidget::updateSigningWarning()
|
|
{
|
|
bool nonRelease = m_step->buildType() != BuildConfiguration::Release;
|
|
bool visible = m_step->signPackage() && nonRelease;
|
|
m_signingDebugWarningLabel->setVisible(visible);
|
|
}
|
|
|
|
// AndroidBuildApkStep
|
|
|
|
AndroidBuildApkStep::AndroidBuildApkStep(BuildStepList *parent, Utils::Id id)
|
|
: AbstractProcessStep(parent, id),
|
|
m_buildTargetSdk(AndroidConfig::apiLevelNameFor(AndroidConfigurations::
|
|
sdkManager()->latestAndroidSdkPlatform()))
|
|
{
|
|
setImmutable(true);
|
|
setDisplayName(Tr::tr("Build Android APK"));
|
|
|
|
connect(this, &BuildStep::addOutput, this, [this](const QString &string, OutputFormat format) {
|
|
if (format == OutputFormat::Stderr)
|
|
stdError(string);
|
|
});
|
|
}
|
|
|
|
bool AndroidBuildApkStep::init()
|
|
{
|
|
if (!AbstractProcessStep::init()) {
|
|
reportWarningOrError(Tr::tr("\"%1\" step failed initialization.").arg(displayName()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (m_signPackage) {
|
|
qCDebug(buildapkstepLog) << "Signing enabled";
|
|
// check keystore and certificate passwords
|
|
if (!verifyKeystorePassword() || !verifyCertificatePassword()) {
|
|
reportWarningOrError(Tr::tr("Keystore/Certificate password verification failed."),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (buildType() != BuildConfiguration::Release)
|
|
reportWarningOrError(Tr::tr("Warning: Signing a debug or profile package."), Task::Warning);
|
|
}
|
|
|
|
QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(kit());
|
|
if (!version) {
|
|
reportWarningOrError(Tr::tr("The Qt version for kit %1 is invalid.").arg(kit()->displayName()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (version->qtVersion() < QVersionNumber(5, 4, 0)) {
|
|
const QString error = Tr::tr("The minimum Qt version required for Gradle build to work is %1. "
|
|
"It is recommended to install the latest Qt version.")
|
|
.arg("5.4.0");
|
|
reportWarningOrError(error, Task::Error);
|
|
return false;
|
|
}
|
|
|
|
const int minSDKForKit = AndroidManager::minimumSDK(kit());
|
|
if (AndroidManager::minimumSDK(target()) < minSDKForKit) {
|
|
const QString error
|
|
= Tr::tr("The API level set for the APK is less than the minimum required by the kit."
|
|
"\nThe minimum API level required by the kit is %1.")
|
|
.arg(minSDKForKit);
|
|
reportWarningOrError(error, Task::Error);
|
|
return false;
|
|
}
|
|
|
|
m_openPackageLocationForRun = m_openPackageLocation;
|
|
const FilePath outputDir = AndroidManager::androidBuildDirectory(target());
|
|
m_packagePath = AndroidManager::packagePath(target());
|
|
|
|
qCDebug(buildapkstepLog).noquote() << "APK or AAB path:" << m_packagePath.toUserOutput();
|
|
|
|
FilePath command = version->hostBinPath().pathAppended("androiddeployqt").withExecutableSuffix();
|
|
|
|
m_inputFile = AndroidQtVersion::androidDeploymentSettings(target());
|
|
if (m_inputFile.isEmpty()) {
|
|
m_skipBuilding = true;
|
|
reportWarningOrError(Tr::tr("No valid input file for \"%1\".").arg(target()->activeBuildKey()),
|
|
Task::Warning);
|
|
return true;
|
|
}
|
|
m_skipBuilding = false;
|
|
|
|
if (m_buildTargetSdk.isEmpty()) {
|
|
reportWarningOrError(Tr::tr("Android build SDK version is not defined. Check Android settings.")
|
|
, Task::Error);
|
|
return false;
|
|
}
|
|
|
|
updateBuildToolsVersionInJsonFile();
|
|
|
|
QStringList arguments = {"--input", m_inputFile.path(),
|
|
"--output", outputDir.path(),
|
|
"--android-platform", m_buildTargetSdk,
|
|
"--jdk", AndroidConfigurations::currentConfig().openJDKLocation().path()};
|
|
|
|
if (m_verbose)
|
|
arguments << "--verbose";
|
|
|
|
arguments << "--gradle";
|
|
|
|
if (m_buildAAB)
|
|
arguments << "--aab" << "--jarsigner";
|
|
|
|
if (buildType() == BuildConfiguration::Release) {
|
|
arguments << "--release";
|
|
}
|
|
|
|
QStringList argumentsPasswordConcealed = arguments;
|
|
|
|
if (m_signPackage) {
|
|
arguments << "--sign" << m_keystorePath.path() << m_certificateAlias
|
|
<< "--storepass" << m_keystorePasswd;
|
|
argumentsPasswordConcealed << "--sign" << "******"
|
|
<< "--storepass" << "******";
|
|
if (!m_certificatePasswd.isEmpty()) {
|
|
arguments << "--keypass" << m_certificatePasswd;
|
|
argumentsPasswordConcealed << "--keypass" << "******";
|
|
}
|
|
|
|
}
|
|
|
|
// Must be the last option, otherwise androiddeployqt might use the other
|
|
// params (e.g. --sign) to choose not to add gdbserver
|
|
if (version->qtVersion() >= QVersionNumber(5, 6, 0)) {
|
|
if (m_addDebugger || buildType() == ProjectExplorer::BuildConfiguration::Debug)
|
|
arguments << "--gdbserver";
|
|
else
|
|
arguments << "--no-gdbserver";
|
|
}
|
|
|
|
processParameters()->setCommandLine({command, arguments});
|
|
|
|
// Generate arguments with keystore password concealed
|
|
setupProcessParameters(&m_concealedParams);
|
|
m_concealedParams.setCommandLine({command, argumentsPasswordConcealed});
|
|
setDisplayedParameters(&m_concealedParams);
|
|
return true;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setupOutputFormatter(OutputFormatter *formatter)
|
|
{
|
|
const auto parser = new JavaParser;
|
|
parser->setProjectFileList(project()->files(Project::AllFiles));
|
|
|
|
const QString buildKey = target()->activeBuildKey();
|
|
const ProjectNode *node = project()->findNodeForBuildKey(buildKey);
|
|
FilePath sourceDirPath;
|
|
if (node)
|
|
sourceDirPath = FilePath::fromVariant(node->data(Constants::AndroidPackageSourceDir));
|
|
parser->setSourceDirectory(sourceDirPath.canonicalPath());
|
|
parser->setBuildDirectory(AndroidManager::androidBuildDirectory(target()));
|
|
formatter->addLineParser(parser);
|
|
AbstractProcessStep::setupOutputFormatter(formatter);
|
|
}
|
|
|
|
void AndroidBuildApkStep::showInGraphicalShell()
|
|
{
|
|
Core::FileUtils::showInGraphicalShell(Core::ICore::dialogParent(), m_packagePath);
|
|
}
|
|
|
|
QWidget *AndroidBuildApkStep::createConfigWidget()
|
|
{
|
|
return new AndroidBuildApkWidget(this);
|
|
}
|
|
|
|
bool AndroidBuildApkStep::verifyKeystorePassword()
|
|
{
|
|
if (!m_keystorePath.exists()) {
|
|
reportWarningOrError(Tr::tr("Cannot sign the package. Invalid keystore path (%1).")
|
|
.arg(m_keystorePath.toUserOutput()), Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (AndroidManager::checkKeystorePassword(m_keystorePath, m_keystorePasswd))
|
|
return true;
|
|
|
|
bool success = false;
|
|
auto verifyCallback = std::bind(&AndroidManager::checkKeystorePassword,
|
|
m_keystorePath, std::placeholders::_1);
|
|
m_keystorePasswd = PasswordInputDialog::getPassword(PasswordInputDialog::KeystorePassword,
|
|
verifyCallback, "", &success);
|
|
return success;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::verifyCertificatePassword()
|
|
{
|
|
if (!AndroidManager::checkCertificateExists(m_keystorePath, m_keystorePasswd,
|
|
m_certificateAlias)) {
|
|
reportWarningOrError(Tr::tr("Cannot sign the package. Certificate alias %1 does not exist.")
|
|
.arg(m_certificateAlias), Task::Error);
|
|
return false;
|
|
}
|
|
|
|
if (AndroidManager::checkCertificatePassword(m_keystorePath, m_keystorePasswd,
|
|
m_certificateAlias, m_certificatePasswd)) {
|
|
return true;
|
|
}
|
|
|
|
bool success = false;
|
|
auto verifyCallback = std::bind(&AndroidManager::checkCertificatePassword,
|
|
m_keystorePath, m_keystorePasswd,
|
|
m_certificateAlias, std::placeholders::_1);
|
|
|
|
m_certificatePasswd = PasswordInputDialog::getPassword(PasswordInputDialog::CertificatePassword,
|
|
verifyCallback, m_certificateAlias,
|
|
&success);
|
|
return success;
|
|
}
|
|
|
|
|
|
static bool copyFileIfNewer(const FilePath &sourceFilePath,
|
|
const FilePath &destinationFilePath)
|
|
{
|
|
if (sourceFilePath == destinationFilePath)
|
|
return true;
|
|
if (destinationFilePath.exists()) {
|
|
if (sourceFilePath.lastModified() <= destinationFilePath.lastModified())
|
|
return true;
|
|
if (!destinationFilePath.removeFile())
|
|
return false;
|
|
}
|
|
|
|
if (!destinationFilePath.parentDir().ensureWritableDir())
|
|
return false;
|
|
expected_str<void> result = sourceFilePath.copyFile(destinationFilePath);
|
|
QTC_ASSERT_EXPECTED(result, return false);
|
|
return true;
|
|
}
|
|
|
|
Tasking::GroupItem AndroidBuildApkStep::runRecipe()
|
|
{
|
|
using namespace Tasking;
|
|
|
|
const auto setupHelper = [this] {
|
|
QtSupport::QtVersion *version = QtSupport::QtKitAspect::qtVersion(kit());
|
|
if (!version) {
|
|
reportWarningOrError(Tr::tr("The Qt version for kit %1 is invalid.")
|
|
.arg(kit()->displayName()), Task::Error);
|
|
return false;
|
|
}
|
|
|
|
const auto androidAbis = AndroidManager::applicationAbis(target());
|
|
const QString buildKey = target()->activeBuildKey();
|
|
const FilePath buildDir = buildDirectory();
|
|
const FilePath androidBuildDir = AndroidManager::androidBuildDirectory(target());
|
|
for (const auto &abi : androidAbis) {
|
|
FilePath androidLibsDir = androidBuildDir / "libs" / abi;
|
|
if (!androidLibsDir.exists()) {
|
|
if (!androidLibsDir.ensureWritableDir()) {
|
|
reportWarningOrError(Tr::tr("The Android build folder %1 was not found and could "
|
|
"not be created.").arg(androidLibsDir.toUserOutput()),
|
|
Task::Error);
|
|
return false;
|
|
} else if (version->qtVersion() >= QVersionNumber(6, 0, 0)
|
|
&& version->qtVersion() <= QVersionNumber(6, 1, 1)) {
|
|
// 6.0.x <= Qt <= 6.1.1 used to need a manaul call to _prepare_apk_dir target,
|
|
// and now it's made directly with ALL target, so this code below ensures
|
|
// these versions are not broken.
|
|
const QString fileName = QString("lib%1_%2.so").arg(buildKey, abi);
|
|
const FilePath from = buildDir / fileName;
|
|
const FilePath to = androidLibsDir / fileName;
|
|
if (!from.exists() || to.exists())
|
|
continue;
|
|
|
|
if (!from.copyFile(to)) {
|
|
reportWarningOrError(Tr::tr("Cannot copy the target's lib file %1 to the "
|
|
"Android build folder %2.")
|
|
.arg(fileName, androidLibsDir.toUserOutput()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const bool inputExists = m_inputFile.exists();
|
|
if (inputExists && !AndroidManager::isQtCreatorGenerated(m_inputFile))
|
|
return true; // use the generated file if it was not generated by qtcreator
|
|
|
|
BuildSystem *bs = buildSystem();
|
|
const FilePaths targets = Utils::transform(
|
|
bs->extraData(buildKey, Android::Constants::AndroidTargets).toStringList(),
|
|
&FilePath::fromUserInput);
|
|
if (targets.isEmpty())
|
|
return inputExists; // qmake does this job for us
|
|
|
|
QJsonObject deploySettings = Android::AndroidManager::deploymentSettings(target());
|
|
QString applicationBinary;
|
|
if (!version->supportsMultipleQtAbis()) {
|
|
QTC_ASSERT(androidAbis.size() == 1, return false);
|
|
applicationBinary = buildSystem()->buildTarget(buildKey).targetFilePath.toString();
|
|
FilePath androidLibsDir = androidBuildDir / "libs" / androidAbis.first();
|
|
for (const FilePath &target : targets) {
|
|
if (!copyFileIfNewer(target, androidLibsDir.pathAppended(target.fileName()))) {
|
|
reportWarningOrError(
|
|
Tr::tr("Cannot copy file \"%1\" to Android build libs folder \"%2\".")
|
|
.arg(target.toUserOutput()).arg(androidLibsDir.toUserOutput()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
}
|
|
deploySettings["target-architecture"] = androidAbis.first();
|
|
} else {
|
|
applicationBinary = buildSystem()->buildTarget(buildKey).targetFilePath.fileName();
|
|
QJsonObject architectures;
|
|
|
|
// Copy targets to android build folder
|
|
for (const auto &abi : androidAbis) {
|
|
QString targetSuffix = QString{"_%1.so"}.arg(abi);
|
|
if (applicationBinary.endsWith(targetSuffix)) {
|
|
// Keep only TargetName from "lib[TargetName]_abi.so"
|
|
applicationBinary.remove(0, 3).chop(targetSuffix.size());
|
|
}
|
|
|
|
FilePath androidLibsDir = androidBuildDir / "libs" / abi;
|
|
for (const FilePath &target : targets) {
|
|
if (target.endsWith(targetSuffix)) {
|
|
const FilePath destination = androidLibsDir.pathAppended(target.fileName());
|
|
if (!copyFileIfNewer(target, destination)) {
|
|
reportWarningOrError(
|
|
Tr::tr("Cannot copy file \"%1\" to Android build libs folder \"%2\".")
|
|
.arg(target.toUserOutput()).arg(androidLibsDir.toUserOutput()),
|
|
Task::Error);
|
|
return false;
|
|
}
|
|
architectures[abi] = AndroidManager::archTriplet(abi);
|
|
}
|
|
}
|
|
}
|
|
deploySettings["architectures"] = architectures;
|
|
}
|
|
deploySettings["application-binary"] = applicationBinary;
|
|
|
|
const QString extraLibs = bs->extraData(buildKey, Android::Constants::AndroidExtraLibs).toString();
|
|
if (!extraLibs.isEmpty())
|
|
deploySettings["android-extra-libs"] = extraLibs;
|
|
|
|
const QString androidSrcs = bs->extraData(buildKey, Android::Constants::AndroidPackageSourceDir).toString();
|
|
if (!androidSrcs.isEmpty())
|
|
deploySettings["android-package-source-directory"] = androidSrcs;
|
|
|
|
const QString qmlImportPath = bs->extraData(buildKey, "QML_IMPORT_PATH").toString();
|
|
if (!qmlImportPath.isEmpty())
|
|
deploySettings["qml-import-paths"] = qmlImportPath;
|
|
|
|
QString qmlRootPath = bs->extraData(buildKey, "QML_ROOT_PATH").toString();
|
|
if (qmlRootPath.isEmpty())
|
|
qmlRootPath = target()->project()->rootProjectDirectory().toString();
|
|
deploySettings["qml-root-path"] = qmlRootPath;
|
|
|
|
QFile f{m_inputFile.toString()};
|
|
if (!f.open(QIODevice::WriteOnly)) {
|
|
reportWarningOrError(Tr::tr("Cannot open androiddeployqt input file \"%1\" for writing.")
|
|
.arg(m_inputFile.toUserOutput()), Task::Error);
|
|
return false;
|
|
}
|
|
f.write(QJsonDocument{deploySettings}.toJson());
|
|
return true;
|
|
};
|
|
|
|
const auto onSetup = [this, setupHelper] {
|
|
if (m_skipBuilding) {
|
|
reportWarningOrError(Tr::tr("Android deploy settings file not found, "
|
|
"not building an APK."), Task::Error);
|
|
return SetupResult::StopWithDone;
|
|
}
|
|
if (AndroidManager::skipInstallationAndPackageSteps(target())) {
|
|
reportWarningOrError(Tr::tr("Product type is not an application, not building an APK."),
|
|
Task::Warning);
|
|
return SetupResult::StopWithDone;
|
|
}
|
|
if (setupHelper())
|
|
return SetupResult::Continue;
|
|
reportWarningOrError(Tr::tr("Cannot set up \"%1\", not building an APK.")
|
|
.arg(displayName()), Task::Error);
|
|
return SetupResult::StopWithError;
|
|
};
|
|
const auto onDone = [this] {
|
|
if (m_openPackageLocationForRun)
|
|
QTimer::singleShot(0, this, &AndroidBuildApkStep::showInGraphicalShell);
|
|
};
|
|
|
|
const Group root {
|
|
onGroupSetup(onSetup),
|
|
onGroupDone(onDone),
|
|
defaultProcessTask()
|
|
};
|
|
return root;
|
|
}
|
|
|
|
void AndroidBuildApkStep::reportWarningOrError(const QString &message, Task::TaskType type)
|
|
{
|
|
qCDebug(buildapkstepLog) << message;
|
|
emit addOutput(message, OutputFormat::ErrorMessage);
|
|
TaskHub::addTask(BuildSystemTask(type, message));
|
|
}
|
|
|
|
void AndroidBuildApkStep::updateBuildToolsVersionInJsonFile()
|
|
{
|
|
expected_str<QByteArray> contents = m_inputFile.fileContents();
|
|
if (!contents)
|
|
return;
|
|
|
|
static const QRegularExpression regex(R"("sdkBuildToolsRevision":."[0-9.]+")");
|
|
QRegularExpressionMatch match = regex.match(QString::fromUtf8(contents.value()));
|
|
const QString version = buildToolsVersion().toString();
|
|
if (match.hasMatch() && !version.isEmpty()) {
|
|
const auto newStr = QLatin1String("\"sdkBuildToolsRevision\": \"%1\"").arg(version).toUtf8();
|
|
contents->replace(match.captured(0).toUtf8(), newStr);
|
|
m_inputFile.writeFileContents(contents.value());
|
|
}
|
|
}
|
|
|
|
void AndroidBuildApkStep::fromMap(const QVariantMap &map)
|
|
{
|
|
m_keystorePath = FilePath::fromSettings(map.value(KeystoreLocationKey));
|
|
m_signPackage = false; // don't restore this
|
|
m_buildTargetSdk = map.value(BuildTargetSdkKey).toString();
|
|
m_buildToolsVersion = QVersionNumber::fromString(map.value(BuildToolsVersionKey).toString());
|
|
if (m_buildTargetSdk.isEmpty()) {
|
|
m_buildTargetSdk = AndroidConfig::apiLevelNameFor(AndroidConfigurations::
|
|
sdkManager()->latestAndroidSdkPlatform());
|
|
}
|
|
m_verbose = map.value(VerboseOutputKey).toBool();
|
|
ProjectExplorer::BuildStep::fromMap(map);
|
|
}
|
|
|
|
void AndroidBuildApkStep::toMap(QVariantMap &map) const
|
|
{
|
|
ProjectExplorer::AbstractProcessStep::toMap(map);
|
|
map.insert(KeystoreLocationKey, m_keystorePath.toSettings());
|
|
map.insert(BuildTargetSdkKey, m_buildTargetSdk);
|
|
map.insert(BuildToolsVersionKey, m_buildToolsVersion.toString());
|
|
map.insert(VerboseOutputKey, m_verbose);
|
|
}
|
|
|
|
Utils::FilePath AndroidBuildApkStep::keystorePath() const
|
|
{
|
|
return m_keystorePath;
|
|
}
|
|
|
|
QString AndroidBuildApkStep::buildTargetSdk() const
|
|
{
|
|
return m_buildTargetSdk;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setBuildTargetSdk(const QString &sdk)
|
|
{
|
|
m_buildTargetSdk = sdk;
|
|
}
|
|
|
|
QVersionNumber AndroidBuildApkStep::buildToolsVersion() const
|
|
{
|
|
return m_buildToolsVersion;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setBuildToolsVersion(const QVersionNumber &version)
|
|
{
|
|
m_buildToolsVersion = version;
|
|
}
|
|
|
|
void AndroidBuildApkStep::stdError(const QString &output)
|
|
{
|
|
QString newOutput = output;
|
|
static const QRegularExpression re("^(\\n)+");
|
|
newOutput.remove(re);
|
|
|
|
if (newOutput.isEmpty())
|
|
return;
|
|
|
|
if (newOutput.startsWith("warning", Qt::CaseInsensitive)
|
|
|| newOutput.startsWith("note", Qt::CaseInsensitive))
|
|
TaskHub::addTask(BuildSystemTask(Task::Warning, newOutput));
|
|
else
|
|
TaskHub::addTask(BuildSystemTask(Task::Error, newOutput));
|
|
}
|
|
|
|
QVariant AndroidBuildApkStep::data(Utils::Id id) const
|
|
{
|
|
if (id == Constants::AndroidNdkPlatform) {
|
|
if (auto qtVersion = QtKitAspect::qtVersion(kit()))
|
|
return AndroidConfigurations::currentConfig()
|
|
.bestNdkPlatformMatch(AndroidManager::minimumSDK(target()), qtVersion);
|
|
return {};
|
|
}
|
|
if (id == Constants::NdkLocation) {
|
|
if (auto qtVersion = QtKitAspect::qtVersion(kit()))
|
|
return QVariant::fromValue(AndroidConfigurations::currentConfig().ndkLocation(qtVersion));
|
|
return {};
|
|
}
|
|
if (id == Constants::SdkLocation)
|
|
return QVariant::fromValue(AndroidConfigurations::currentConfig().sdkLocation());
|
|
|
|
if (id == Constants::AndroidMkSpecAbis)
|
|
return AndroidManager::applicationAbis(target());
|
|
|
|
return AbstractProcessStep::data(id);
|
|
}
|
|
|
|
void AndroidBuildApkStep::setKeystorePath(const Utils::FilePath &path)
|
|
{
|
|
m_keystorePath = path;
|
|
m_certificatePasswd.clear();
|
|
m_keystorePasswd.clear();
|
|
}
|
|
|
|
void AndroidBuildApkStep::setKeystorePassword(const QString &pwd)
|
|
{
|
|
m_keystorePasswd = pwd;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setCertificateAlias(const QString &alias)
|
|
{
|
|
m_certificateAlias = alias;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setCertificatePassword(const QString &pwd)
|
|
{
|
|
m_certificatePasswd = pwd;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::signPackage() const
|
|
{
|
|
return m_signPackage;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setSignPackage(bool b)
|
|
{
|
|
m_signPackage = b;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::buildAAB() const
|
|
{
|
|
return m_buildAAB;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setBuildAAB(bool aab)
|
|
{
|
|
m_buildAAB = aab;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::openPackageLocation() const
|
|
{
|
|
return m_openPackageLocation;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setOpenPackageLocation(bool open)
|
|
{
|
|
m_openPackageLocation = open;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setVerboseOutput(bool verbose)
|
|
{
|
|
m_verbose = verbose;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::addDebugger() const
|
|
{
|
|
return m_addDebugger;
|
|
}
|
|
|
|
void AndroidBuildApkStep::setAddDebugger(bool debug)
|
|
{
|
|
m_addDebugger = debug;
|
|
}
|
|
|
|
bool AndroidBuildApkStep::verboseOutput() const
|
|
{
|
|
return m_verbose;
|
|
}
|
|
|
|
QAbstractItemModel *AndroidBuildApkStep::keystoreCertificates()
|
|
{
|
|
// check keystore passwords
|
|
if (!verifyKeystorePassword())
|
|
return nullptr;
|
|
|
|
CertificatesModel *model = nullptr;
|
|
const QStringList params = {"-list", "-v", "-keystore", m_keystorePath.toUserOutput(),
|
|
"-storepass", m_keystorePasswd, "-J-Duser.language=en"};
|
|
|
|
Process keytoolProc;
|
|
keytoolProc.setTimeoutS(30);
|
|
keytoolProc.setCommand({AndroidConfigurations::currentConfig().keytoolPath(), params});
|
|
keytoolProc.runBlocking(EventLoopMode::On);
|
|
if (keytoolProc.result() > ProcessResult::FinishedWithError)
|
|
QMessageBox::critical(nullptr, Tr::tr("Error"), Tr::tr("Failed to run keytool."));
|
|
else
|
|
model = new CertificatesModel(keytoolProc.cleanedStdOut(), this);
|
|
|
|
return model;
|
|
}
|
|
|
|
PasswordInputDialog::PasswordInputDialog(PasswordInputDialog::Context context,
|
|
std::function<bool (const QString &)> callback,
|
|
const QString &extraContextStr,
|
|
QWidget *parent) :
|
|
QDialog(parent, Qt::Dialog | Qt::CustomizeWindowHint | Qt::WindowTitleHint),
|
|
verifyCallback(callback)
|
|
|
|
{
|
|
inputEdit->setEchoMode(QLineEdit::Password);
|
|
|
|
warningLabel->hide();
|
|
|
|
auto mainLayout = new QVBoxLayout(this);
|
|
mainLayout->addWidget(inputContextlabel);
|
|
mainLayout->addWidget(inputEdit);
|
|
mainLayout->addWidget(warningLabel);
|
|
mainLayout->addWidget(buttonBox);
|
|
|
|
connect(inputEdit, &QLineEdit::textChanged, this, [this](const QString &text) {
|
|
buttonBox->button(QDialogButtonBox::Ok)->setEnabled(!text.isEmpty());
|
|
});
|
|
|
|
connect(buttonBox, &QDialogButtonBox::accepted, this, [this] {
|
|
if (verifyCallback(inputEdit->text())) {
|
|
accept(); // Dialog accepted.
|
|
} else {
|
|
warningLabel->show();
|
|
inputEdit->clear();
|
|
adjustSize();
|
|
}
|
|
});
|
|
|
|
connect(buttonBox, &QDialogButtonBox::rejected, this, &QDialog::reject);
|
|
|
|
setWindowTitle(context == KeystorePassword ? Tr::tr("Keystore") : Tr::tr("Certificate"));
|
|
|
|
QString contextStr;
|
|
if (context == KeystorePassword)
|
|
contextStr = Tr::tr("Enter keystore password");
|
|
else
|
|
contextStr = Tr::tr("Enter certificate password");
|
|
|
|
contextStr += extraContextStr.isEmpty() ? QStringLiteral(":") :
|
|
QStringLiteral(" (%1):").arg(extraContextStr);
|
|
inputContextlabel->setText(contextStr);
|
|
}
|
|
|
|
QString PasswordInputDialog::getPassword(Context context, std::function<bool (const QString &)> callback,
|
|
const QString &extraContextStr, bool *ok, QWidget *parent)
|
|
{
|
|
std::unique_ptr<PasswordInputDialog> dlg(new PasswordInputDialog(context, callback,
|
|
extraContextStr, parent));
|
|
bool isAccepted = dlg->exec() == QDialog::Accepted;
|
|
if (ok)
|
|
*ok = isAccepted;
|
|
return isAccepted ? dlg->inputEdit->text() : "";
|
|
}
|
|
|
|
|
|
// AndroidBuildApkStepFactory
|
|
|
|
AndroidBuildApkStepFactory::AndroidBuildApkStepFactory()
|
|
{
|
|
registerStep<AndroidBuildApkStep>(Constants::ANDROID_BUILD_APK_ID);
|
|
setSupportedDeviceType(Constants::ANDROID_DEVICE_TYPE);
|
|
setSupportedStepList(ProjectExplorer::Constants::BUILDSTEPS_BUILD);
|
|
setDisplayName(Tr::tr("Build Android APK"));
|
|
setRepeatable(false);
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace Android
|