Android: Rework BuildApkWidget

Merge BuildApkInnerWidget into BuildApkWidget and simplify
layout structure. The inner .ui managed three of the groupboxes,
the outer added two more manually.

Change-Id: If4b54fa569952a9bb5eedb889a271827d8a88577
Reviewed-by: Leena Miettinen <riitta-leena.miettinen@qt.io>
Reviewed-by: BogDan Vatra <bogdan@kdab.com>
This commit is contained in:
hjk
2019-01-25 16:25:55 +01:00
parent 0eb3b54434
commit 357ecd9297
5 changed files with 266 additions and 466 deletions

View File

@@ -101,7 +101,6 @@ FORMS += \
addnewavddialog.ui \ addnewavddialog.ui \
androidcreatekeystorecertificate.ui \ androidcreatekeystorecertificate.ui \
androiddevicedialog.ui \ androiddevicedialog.ui \
androidbuildapkwidget.ui \
adbcommandswidget.ui \ adbcommandswidget.ui \
androidsdkmanagerwidget.ui androidsdkmanagerwidget.ui

View File

@@ -36,7 +36,6 @@ Project {
"androidbuildapkstep.h", "androidbuildapkstep.h",
"androidbuildapkwidget.cpp", "androidbuildapkwidget.cpp",
"androidbuildapkwidget.h", "androidbuildapkwidget.h",
"androidbuildapkwidget.ui",
"androiddeployqtstep.cpp", "androiddeployqtstep.cpp",
"androiddeployqtstep.h", "androiddeployqtstep.h",
"androiddebugsupport.cpp", "androiddebugsupport.cpp",

View File

@@ -24,14 +24,15 @@
** **
****************************************************************************/ ****************************************************************************/
#include "androidbuildapkstep.h"
#include "androidbuildapkwidget.h" #include "androidbuildapkwidget.h"
#include "androidbuildapkstep.h"
#include "androidconfigurations.h" #include "androidconfigurations.h"
#include "androidextralibrarylistmodel.h"
#include "androidcreatekeystorecertificate.h" #include "androidcreatekeystorecertificate.h"
#include "androidmanager.h" #include "androidmanager.h"
#include "androidsdkmanager.h" #include "androidsdkmanager.h"
#include "createandroidmanifestwizard.h" #include "createandroidmanifestwizard.h"
#include "ui_androidbuildapkwidget.h"
#include <projectexplorer/buildconfiguration.h> #include <projectexplorer/buildconfiguration.h>
#include <projectexplorer/project.h> #include <projectexplorer/project.h>
@@ -45,8 +46,10 @@
#include <utils/pathchooser.h> #include <utils/pathchooser.h>
#include <utils/utilsicons.h> #include <utils/utilsicons.h>
#include <QFileDialog> #include <QCheckBox>
#include <QComboBox>
#include <QGroupBox> #include <QGroupBox>
#include <QFileDialog>
#include <QLabel> #include <QLabel>
#include <QListView> #include <QListView>
#include <QPushButton> #include <QPushButton>
@@ -56,100 +59,275 @@
#include <algorithm> #include <algorithm>
using namespace ProjectExplorer; using namespace ProjectExplorer;
using namespace Utils;
namespace Android { namespace Android {
namespace Internal { namespace Internal {
AndroidBuildApkInnerWidget::AndroidBuildApkInnerWidget(AndroidBuildApkStep *step) AndroidBuildApkWidget::AndroidBuildApkWidget(AndroidBuildApkStep *step)
: ProjectExplorer::BuildStepConfigWidget(step), : BuildStepConfigWidget(step), m_step(step)
m_ui(new Ui::AndroidBuildApkWidget),
m_step(step)
{ {
m_ui->setupUi(this);
setDisplayName("<b>" + tr("Build Android APK") + "</b>"); setDisplayName("<b>" + tr("Build Android APK") + "</b>");
setSummaryText(displayName()); setSummaryText(displayName());
// Target sdk combobox auto vbox = new QVBoxLayout(this);
vbox->addWidget(createSignPackageGroup());
vbox->addWidget(createApplicationGroup());
vbox->addWidget(createAdvancedGroup());
vbox->addWidget(createCreateTemplatesGroup());
vbox->addWidget(createAdditionalLibrariesGroup());
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()
{
const int minApiSupported = AndroidManager::apiLevelRange().first; const int minApiSupported = AndroidManager::apiLevelRange().first;
QStringList targets = AndroidConfig::apiLevelNamesFor(AndroidConfigurations::sdkManager()-> QStringList targets = AndroidConfig::apiLevelNamesFor(AndroidConfigurations::sdkManager()->
filteredSdkPlatforms(minApiSupported)); filteredSdkPlatforms(minApiSupported));
targets.removeDuplicates(); targets.removeDuplicates();
m_ui->targetSDKComboBox->addItems(targets);
m_ui->targetSDKComboBox->setCurrentIndex(targets.indexOf(AndroidManager::buildTargetSDK(step->target())));
// Ministro auto group = new QGroupBox(tr("Application"), this);
if (m_step->useMinistro())
m_ui->ministroOption->setChecked(true);
// signing auto targetSDKComboBox = new QComboBox(group);
m_ui->signPackageCheckBox->setChecked(m_step->signPackage()); targetSDKComboBox->addItems(targets);
m_ui->KeystoreLocationPathChooser->setExpectedKind(Utils::PathChooser::File); targetSDKComboBox->setCurrentIndex(targets.indexOf(AndroidManager::buildTargetSDK(step()->target())));
m_ui->KeystoreLocationPathChooser->lineEdit()->setReadOnly(true);
m_ui->KeystoreLocationPathChooser->setPath(m_step->keystorePath().toUserOutput());
m_ui->KeystoreLocationPathChooser->setInitialBrowsePathBackup(QDir::homePath());
m_ui->KeystoreLocationPathChooser->setPromptDialogFilter(tr("Keystore files (*.keystore *.jks)"));
m_ui->KeystoreLocationPathChooser->setPromptDialogTitle(tr("Select Keystore File"));
m_ui->signingDebugWarningIcon->setPixmap(Utils::Icons::WARNING.pixmap());
m_ui->signingDebugWarningIcon->hide();
m_ui->signingDebugWarningLabel->hide();
signPackageCheckBoxToggled(m_step->signPackage());
m_ui->verboseOutputCheckBox->setChecked(m_step->verboseOutput());
m_ui->openPackageLocationCheckBox->setChecked(m_step->openPackageLocation());
m_ui->addDebuggerCheckBox->setChecked(m_step->addDebugger());
// target sdk
const auto cbActivated = QOverload<int>::of(&QComboBox::activated); const auto cbActivated = QOverload<int>::of(&QComboBox::activated);
const auto cbCurrentIndexChanged = QOverload<int>::of(&QComboBox::currentIndexChanged); connect(targetSDKComboBox, cbActivated, this, [this, targetSDKComboBox](int idx) {
connect(m_ui->targetSDKComboBox, cbActivated, this, [this](int idx) { const QString sdk = targetSDKComboBox->itemText(idx);
const QString sdk = m_ui->targetSDKComboBox->itemText(idx);
m_step->setBuildTargetSdk(sdk); m_step->setBuildTargetSdk(sdk);
}); });
// deployment options auto hbox = new QHBoxLayout(group);
connect(m_ui->ministroOption, &QAbstractButton::clicked, hbox->addWidget(new QLabel(tr("Android build SDK:"), group));
m_step, &AndroidBuildApkStep::setUseMinistro); hbox->addWidget(targetSDKComboBox);
connect(m_ui->openPackageLocationCheckBox, &QAbstractButton::toggled, return group;
this, &AndroidBuildApkInnerWidget::openPackageLocationCheckBoxToggled); }
connect(m_ui->verboseOutputCheckBox, &QAbstractButton::toggled,
this, &AndroidBuildApkInnerWidget::verboseOutputCheckBoxToggled);
connect(m_ui->addDebuggerCheckBox, &QAbstractButton::toggled,
m_step, &AndroidBuildApkStep::setAddDebugger);
//signing QWidget *AndroidBuildApkWidget::createSignPackageGroup()
connect(m_ui->signPackageCheckBox, &QAbstractButton::toggled, {
this, &AndroidBuildApkInnerWidget::signPackageCheckBoxToggled); QSizePolicy sizePolicy(QSizePolicy::Fixed, QSizePolicy::Preferred);
connect(m_ui->KeystoreCreatePushButton, &QAbstractButton::clicked, sizePolicy.setHorizontalStretch(0);
this, &AndroidBuildApkInnerWidget::createKeyStore); sizePolicy.setVerticalStretch(0);
connect(m_ui->KeystoreLocationPathChooser, &Utils::PathChooser::pathChanged,
this, &AndroidBuildApkInnerWidget::updateKeyStorePath); auto group = new QGroupBox(tr("Sign package"), this);
auto keystoreLocationLabel = new QLabel(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->setPath(m_step->keystorePath().toUserOutput());
keystoreLocationChooser->setInitialBrowsePathBackup(QDir::homePath());
keystoreLocationChooser->setPromptDialogFilter(tr("Keystore files (*.keystore *.jks)"));
keystoreLocationChooser->setPromptDialogTitle(tr("Select Keystore File"));
connect(keystoreLocationChooser, &PathChooser::pathChanged, this, [this](const QString &path) {
FileName file = FileName::fromString(path);
m_step->setKeystorePath(file);
m_signPackageCheckBox->setChecked(!file.isEmpty());
if (!file.isEmpty())
setCertificates();
});
auto keystoreCreateButton = new QPushButton(tr("Create..."), group);
connect(keystoreCreateButton, &QAbstractButton::clicked, this, [this, keystoreLocationChooser] {
AndroidCreateKeystoreCertificate d;
if (d.exec() != QDialog::Accepted)
return;
keystoreLocationChooser->setPath(d.keystoreFilePath().toUserOutput());
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("Sign package"), group);
m_signPackageCheckBox->setChecked(m_step->signPackage());
m_signingDebugWarningIcon = new QLabel(group);
m_signingDebugWarningIcon->setSizePolicy(sizePolicy);
m_signingDebugWarningIcon->setPixmap(Icons::WARNING.pixmap());
m_signingDebugWarningIcon->hide();
m_signingDebugWarningLabel = new QLabel(tr("Signing a debug package"), group);
m_signingDebugWarningLabel->setSizePolicy(sizePolicy);
m_signingDebugWarningLabel->hide();
auto certificateAliasLabel = new QLabel(tr("Certificate alias:"), group);
certificateAliasLabel->setAlignment(Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter);
m_certificatesAliasComboBox = new QComboBox(group);
m_certificatesAliasComboBox->setEnabled(false);
QSizePolicy sizePolicy2(QSizePolicy::Fixed, QSizePolicy::Fixed);
sizePolicy2.setHorizontalStretch(0);
sizePolicy2.setVerticalStretch(0);
m_certificatesAliasComboBox->setSizePolicy(sizePolicy2);
m_certificatesAliasComboBox->setMinimumSize(QSize(300, 0));
auto horizontalLayout_2 = new QHBoxLayout;
horizontalLayout_2->addWidget(keystoreLocationLabel);
horizontalLayout_2->addWidget(keystoreLocationChooser);
horizontalLayout_2->addWidget(keystoreCreateButton);
auto horizontalLayout_3 = new QHBoxLayout;
horizontalLayout_3->addWidget(m_signingDebugWarningIcon);
horizontalLayout_3->addWidget(m_signingDebugWarningLabel);
horizontalLayout_3->addWidget(certificateAliasLabel);
horizontalLayout_3->addWidget(m_certificatesAliasComboBox);
auto vbox = new QVBoxLayout(group);
vbox->addLayout(horizontalLayout_2);
vbox->addWidget(m_signPackageCheckBox);
vbox->addLayout(horizontalLayout_3);
connect(m_signPackageCheckBox, &QAbstractButton::toggled,
this, &AndroidBuildApkWidget::signPackageCheckBoxToggled);
auto updateAlias = [this](int idx) { auto updateAlias = [this](int idx) {
QString alias = m_ui->certificatesAliasComboBox->itemText(idx); QString alias = m_certificatesAliasComboBox->itemText(idx);
if (alias.length()) if (alias.length())
m_step->setCertificateAlias(alias); m_step->setCertificateAlias(alias);
}; };
connect(m_ui->certificatesAliasComboBox, cbActivated, this, updateAlias); const auto cbActivated = QOverload<int>::of(&QComboBox::activated);
connect(m_ui->certificatesAliasComboBox, cbCurrentIndexChanged, this, updateAlias); const auto cbCurrentIndexChanged = QOverload<int>::of(&QComboBox::currentIndexChanged);
connect(m_step->buildConfiguration(), &ProjectExplorer::BuildConfiguration::buildTypeChanged, connect(m_certificatesAliasComboBox, cbActivated, this, updateAlias);
this, &AndroidBuildApkInnerWidget::updateSigningWarning); connect(m_certificatesAliasComboBox, cbCurrentIndexChanged, this, updateAlias);
updateSigningWarning(); return group;
} }
AndroidBuildApkInnerWidget::~AndroidBuildApkInnerWidget() QWidget *AndroidBuildApkWidget::createAdvancedGroup()
{ {
delete m_ui; auto group = new QGroupBox(tr("Advanced Actions"), this);
auto openPackageLocationCheckBox = new QCheckBox(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("Add debug server"), group);
m_addDebuggerCheckBox->setEnabled(false);
m_addDebuggerCheckBox->setToolTip(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("Verbose output"), group);
verboseOutputCheckBox->setChecked(m_step->verboseOutput());
auto ministroOption = new QCheckBox(tr("Use Ministro service to install Qt"), group);
ministroOption->setToolTip(tr("Uses the external Ministro application to download and maintain Qt libraries."));
ministroOption->setChecked(m_step->useMinistro());
connect(ministroOption, &QAbstractButton::clicked,
m_step, &AndroidBuildApkStep::setUseMinistro);
auto vbox = new QVBoxLayout(group);
vbox->addWidget(openPackageLocationCheckBox);
vbox->addWidget(verboseOutputCheckBox);
vbox->addWidget(m_addDebuggerCheckBox);
vbox->addWidget(ministroOption);
connect(verboseOutputCheckBox, &QAbstractButton::toggled,
this, [this](bool checked) { m_step->setVerboseOutput(checked); });
return group;
} }
void AndroidBuildApkInnerWidget::signPackageCheckBoxToggled(bool checked) QWidget *AndroidBuildApkWidget::createCreateTemplatesGroup()
{ {
m_ui->certificatesAliasComboBox->setEnabled(checked); auto createTemplatesGroupBox = new QGroupBox(tr("Android"));
createTemplatesGroupBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
auto createAndroidTemplatesButton = new QPushButton(tr("Create Templates"));
connect(createAndroidTemplatesButton, &QAbstractButton::clicked, this, [this] {
CreateAndroidManifestWizard wizard(m_step->target());
wizard.exec();
});
auto horizontalLayout = new QHBoxLayout(createTemplatesGroupBox);
horizontalLayout->addWidget(createAndroidTemplatesButton);
horizontalLayout->addStretch(1);
return createTemplatesGroupBox;
}
QWidget *AndroidBuildApkWidget::createAdditionalLibrariesGroup()
{
auto group = new QGroupBox(tr("Additional Libraries"));
group->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
auto libsModel = new AndroidExtraLibraryListModel(m_step->target(), this);
connect(libsModel, &AndroidExtraLibraryListModel::enabledChanged,
group, &QWidget::setEnabled);
auto libsView = new QListView;
libsView->setSelectionMode(QAbstractItemView::ExtendedSelection);
libsView->setToolTip(tr("List of extra libraries to include in Android package and load on startup."));
libsView->setModel(libsModel);
auto addLibButton = new QToolButton;
addLibButton->setText(tr("Add..."));
addLibButton->setToolTip(tr("Select library to include in package."));
addLibButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
addLibButton->setToolButtonStyle(Qt::ToolButtonTextOnly);
connect(addLibButton, &QAbstractButton::clicked, this, [this, libsModel] {
QStringList fileNames = QFileDialog::getOpenFileNames(this,
tr("Select additional libraries"),
QDir::homePath(),
tr("Libraries (*.so)"));
if (!fileNames.isEmpty())
libsModel->addEntries(fileNames);
});
auto removeLibButton = new QToolButton;
removeLibButton->setText(tr("Remove"));
removeLibButton->setToolTip(tr("Remove currently selected library from list."));
connect(removeLibButton, &QAbstractButton::clicked, this, [libsModel, libsView] {
QModelIndexList removeList = libsView->selectionModel()->selectedIndexes();
libsModel->removeEntries(removeList);
});
auto libsButtonLayout = new QVBoxLayout;
libsButtonLayout->addWidget(addLibButton);
libsButtonLayout->addWidget(removeLibButton);
libsButtonLayout->addStretch(1);
auto hbox = new QHBoxLayout(group);
hbox->addWidget(libsView);
hbox->addLayout(libsButtonLayout);
QItemSelectionModel *libSelection = libsView->selectionModel();
connect(libSelection, &QItemSelectionModel::selectionChanged, this, [libSelection, removeLibButton] {
removeLibButton->setEnabled(libSelection->hasSelection());
});
Target *target = m_step->target();
RunConfiguration *rc = target->activeRunConfiguration();
const ProjectNode *node = rc ? target->project()->findNodeForBuildKey(rc->buildKey()) : nullptr;
group->setEnabled(node && !node->parseInProgress());
return group;
}
void AndroidBuildApkWidget::signPackageCheckBoxToggled(bool checked)
{
m_certificatesAliasComboBox->setEnabled(checked);
m_step->setSignPackage(checked); m_step->setSignPackage(checked);
m_ui->addDebuggerCheckBox->setChecked(!checked); m_addDebuggerCheckBox->setChecked(!checked);
updateSigningWarning(); updateSigningWarning();
if (!checked) if (!checked)
return; return;
@@ -157,159 +335,21 @@ void AndroidBuildApkInnerWidget::signPackageCheckBoxToggled(bool checked)
setCertificates(); setCertificates();
} }
void AndroidBuildApkInnerWidget::createKeyStore() void AndroidBuildApkWidget::setCertificates()
{
AndroidCreateKeystoreCertificate d;
if (d.exec() != QDialog::Accepted)
return;
m_ui->KeystoreLocationPathChooser->setPath(d.keystoreFilePath().toUserOutput());
m_step->setKeystorePath(d.keystoreFilePath());
m_step->setKeystorePassword(d.keystorePassword());
m_step->setCertificateAlias(d.certificateAlias());
m_step->setCertificatePassword(d.certificatePassword());
setCertificates();
}
void AndroidBuildApkInnerWidget::setCertificates()
{ {
QAbstractItemModel *certificates = m_step->keystoreCertificates(); QAbstractItemModel *certificates = m_step->keystoreCertificates();
if (certificates) { if (certificates) {
m_ui->signPackageCheckBox->setChecked(certificates); m_signPackageCheckBox->setChecked(certificates);
m_ui->certificatesAliasComboBox->setModel(certificates); m_certificatesAliasComboBox->setModel(certificates);
} }
} }
void AndroidBuildApkInnerWidget::updateKeyStorePath(const QString &path) void AndroidBuildApkWidget::updateSigningWarning()
{ {
Utils::FileName file = Utils::FileName::fromString(path); bool nonRelease = m_step->buildConfiguration()->buildType() != BuildConfiguration::Release;
m_step->setKeystorePath(file); bool visible = m_step->signPackage() && nonRelease;
m_ui->signPackageCheckBox->setChecked(!file.isEmpty()); m_signingDebugWarningIcon->setVisible(visible);
if (!file.isEmpty()) m_signingDebugWarningLabel->setVisible(visible);
setCertificates();
}
void AndroidBuildApkInnerWidget::openPackageLocationCheckBoxToggled(bool checked)
{
m_step->setOpenPackageLocation(checked);
}
void AndroidBuildApkInnerWidget::verboseOutputCheckBoxToggled(bool checked)
{
m_step->setVerboseOutput(checked);
}
void AndroidBuildApkInnerWidget::updateSigningWarning()
{
bool nonRelease = m_step->buildConfiguration()->buildType()
!= ProjectExplorer::BuildConfiguration::Release;
if (m_step->signPackage() && nonRelease) {
m_ui->signingDebugWarningIcon->setVisible(true);
m_ui->signingDebugWarningLabel->setVisible(true);
} else {
m_ui->signingDebugWarningIcon->setVisible(false);
m_ui->signingDebugWarningLabel->setVisible(false);
}
}
// AndroidBuildApkWidget
AndroidBuildApkWidget::AndroidBuildApkWidget(AndroidBuildApkStep *step) :
BuildStepConfigWidget(step),
m_step(step)
{
setDisplayName("<b>" + tr("Build Android APK") + "</b>");
setSummaryText("<b>" + tr("Build Android APK") + "</b>");
m_extraLibraryListModel = new AndroidExtraLibraryListModel(m_step->target(), this);
auto base = new AndroidBuildApkInnerWidget(step);
base->layout()->setContentsMargins(0, 0, 0, 0);
auto createTemplatesGroupBox = new QGroupBox(tr("Android"));
createTemplatesGroupBox->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
auto createAndroidTemplatesButton = new QPushButton(tr("Create Templates"));
auto horizontalLayout = new QHBoxLayout(createTemplatesGroupBox);
horizontalLayout->addWidget(createAndroidTemplatesButton);
horizontalLayout->addStretch(1);
auto additionalLibrariesGroupBox = new QGroupBox(tr("Additional Libraries"));
additionalLibrariesGroupBox->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
m_androidExtraLibsListView = new QListView;
m_androidExtraLibsListView->setSelectionMode(QAbstractItemView::ExtendedSelection);
m_androidExtraLibsListView->setToolTip(tr("List of extra libraries to include in Android package and load on startup."));
m_androidExtraLibsListView->setModel(m_extraLibraryListModel);
auto addAndroidExtraLibButton = new QToolButton;
addAndroidExtraLibButton->setText(tr("Add..."));
addAndroidExtraLibButton->setToolTip(tr("Select library to include in package."));
addAndroidExtraLibButton->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
addAndroidExtraLibButton->setToolButtonStyle(Qt::ToolButtonTextOnly);
m_removeAndroidExtraLibButton = new QToolButton;
m_removeAndroidExtraLibButton->setText(tr("Remove"));
m_removeAndroidExtraLibButton->setToolTip(tr("Remove currently selected library from list."));
auto androidExtraLibsButtonLayout = new QVBoxLayout();
androidExtraLibsButtonLayout->addWidget(addAndroidExtraLibButton);
androidExtraLibsButtonLayout->addWidget(m_removeAndroidExtraLibButton);
androidExtraLibsButtonLayout->addStretch(1);
auto androidExtraLibsLayout = new QHBoxLayout(additionalLibrariesGroupBox);
androidExtraLibsLayout->addWidget(m_androidExtraLibsListView);
androidExtraLibsLayout->addLayout(androidExtraLibsButtonLayout);
auto topLayout = new QVBoxLayout(this);
topLayout->addWidget(base);
topLayout->addWidget(createTemplatesGroupBox);
topLayout->addWidget(additionalLibrariesGroupBox);
connect(createAndroidTemplatesButton, &QAbstractButton::clicked, this, [this] {
CreateAndroidManifestWizard wizard(m_step->target());
wizard.exec();
});
connect(addAndroidExtraLibButton, &QAbstractButton::clicked,
this, &AndroidBuildApkWidget::addAndroidExtraLib);
connect(m_removeAndroidExtraLibButton, &QAbstractButton::clicked,
this, &AndroidBuildApkWidget::removeAndroidExtraLib);
connect(m_androidExtraLibsListView->selectionModel(), &QItemSelectionModel::selectionChanged,
this, &AndroidBuildApkWidget::checkEnableRemoveButton);
connect(m_extraLibraryListModel, &AndroidExtraLibraryListModel::enabledChanged,
additionalLibrariesGroupBox, &QWidget::setEnabled);
Target *target = m_step->target();
RunConfiguration *rc = target->activeRunConfiguration();
const ProjectNode *node = rc ? target->project()->findNodeForBuildKey(rc->buildKey()) : nullptr;
additionalLibrariesGroupBox->setEnabled(node && !node->parseInProgress());
}
void AndroidBuildApkWidget::addAndroidExtraLib()
{
QStringList fileNames = QFileDialog::getOpenFileNames(this,
tr("Select additional libraries"),
QDir::homePath(),
tr("Libraries (*.so)"));
if (!fileNames.isEmpty())
m_extraLibraryListModel->addEntries(fileNames);
}
void AndroidBuildApkWidget::removeAndroidExtraLib()
{
QModelIndexList removeList = m_androidExtraLibsListView->selectionModel()->selectedIndexes();
m_extraLibraryListModel->removeEntries(removeList);
}
void AndroidBuildApkWidget::checkEnableRemoveButton()
{
m_removeAndroidExtraLibButton->setEnabled(m_androidExtraLibsListView->selectionModel()->hasSelection());
} }
} // Internal } // Internal

View File

@@ -29,42 +29,18 @@
#include "android_global.h" #include "android_global.h"
#include "androidbuildapkstep.h" #include "androidbuildapkstep.h"
#include "androidextralibrarylistmodel.h"
#include <projectexplorer/buildstep.h> #include <projectexplorer/buildstep.h>
#include <QListView>
#include <QToolButton>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
namespace Ui { class AndroidBuildApkWidget; } class QCheckBox;
class QComboBox;
class QLabel;
QT_END_NAMESPACE QT_END_NAMESPACE
namespace Android { namespace Android {
namespace Internal { namespace Internal {
class AndroidBuildApkInnerWidget : public ProjectExplorer::BuildStepConfigWidget
{
Q_OBJECT
public:
AndroidBuildApkInnerWidget(AndroidBuildApkStep *step);
~AndroidBuildApkInnerWidget() override;
private:
void createKeyStore();
void updateSigningWarning();
void openPackageLocationCheckBoxToggled(bool checked);
void verboseOutputCheckBoxToggled(bool checked);
void updateKeyStorePath(const QString &path);
void signPackageCheckBoxToggled(bool checked);
void setCertificates();
Ui::AndroidBuildApkWidget *m_ui;
AndroidBuildApkStep *m_step;
};
class AndroidBuildApkWidget : public ProjectExplorer::BuildStepConfigWidget class AndroidBuildApkWidget : public ProjectExplorer::BuildStepConfigWidget
{ {
Q_OBJECT Q_OBJECT
@@ -72,21 +48,24 @@ class AndroidBuildApkWidget : public ProjectExplorer::BuildStepConfigWidget
public: public:
explicit AndroidBuildApkWidget(AndroidBuildApkStep *step); explicit AndroidBuildApkWidget(AndroidBuildApkStep *step);
signals: private:
void requestAndroidTemplates(); void setCertificates();
void updateSigningWarning();
void signPackageCheckBoxToggled(bool checked);
QWidget *createApplicationGroup();
QWidget *createSignPackageGroup();
QWidget *createAdvancedGroup();
QWidget *createCreateTemplatesGroup();
QWidget *createAdditionalLibrariesGroup();
private: private:
void addAndroidExtraLib();
void removeAndroidExtraLib();
void checkEnableRemoveButton();
private:
QListView *m_androidExtraLibsListView = nullptr;
QToolButton *m_removeAndroidExtraLibButton = nullptr;
AndroidBuildApkStep *m_step = nullptr; AndroidBuildApkStep *m_step = nullptr;
Android::AndroidExtraLibraryListModel *m_extraLibraryListModel = nullptr; QCheckBox *m_signPackageCheckBox = nullptr;
bool m_ignoreChange = false; QLabel *m_signingDebugWarningIcon = nullptr;
QLabel *m_signingDebugWarningLabel = nullptr;
QComboBox *m_certificatesAliasComboBox = nullptr;
QCheckBox *m_addDebuggerCheckBox = nullptr;
}; };
} // namespace Internal } // namespace Internal

View File

@@ -1,217 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
<class>AndroidBuildApkWidget</class>
<widget class="QWidget" name="AndroidBuildApkWidget">
<property name="geometry">
<rect>
<x>0</x>
<y>0</y>
<width>641</width>
<height>349</height>
</rect>
</property>
<property name="windowTitle">
<string/>
</property>
<layout class="QGridLayout" name="gridLayout_2">
<item row="1" column="0" colspan="2">
<widget class="QGroupBox" name="signPackage">
<property name="title">
<string>Sign package</string>
</property>
<layout class="QVBoxLayout" name="verticalLayout_2">
<item>
<layout class="QHBoxLayout" name="horizontalLayout_2">
<item>
<widget class="QLabel" name="KeystoreLocationLabel">
<property name="text">
<string>Keystore:</string>
</property>
<property name="alignment">
<set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
</property>
</widget>
</item>
<item>
<widget class="Utils::PathChooser" name="KeystoreLocationPathChooser" native="true"/>
</item>
<item>
<widget class="QPushButton" name="KeystoreCreatePushButton">
<property name="text">
<string>Create...</string>
</property>
</widget>
</item>
</layout>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout_4">
<item>
<widget class="QCheckBox" name="signPackageCheckBox">
<property name="text">
<string>Sign package</string>
</property>
</widget>
</item>
</layout>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout_3">
<item>
<widget class="QLabel" name="signingDebugWarningIcon">
<property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="signingDebugWarningLabel">
<property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Preferred">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="text">
<string>Signing a debug package</string>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="KeystoreLocationLabel_2">
<property name="sizePolicy">
<sizepolicy hsizetype="Preferred" vsizetype="Preferred">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="text">
<string>Certificate alias:</string>
</property>
<property name="alignment">
<set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
</property>
</widget>
</item>
<item>
<widget class="QComboBox" name="certificatesAliasComboBox">
<property name="enabled">
<bool>false</bool>
</property>
<property name="sizePolicy">
<sizepolicy hsizetype="Fixed" vsizetype="Fixed">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="minimumSize">
<size>
<width>300</width>
<height>0</height>
</size>
</property>
</widget>
</item>
</layout>
</item>
</layout>
</widget>
</item>
<item row="0" column="0" colspan="2">
<widget class="QGroupBox" name="application">
<property name="title">
<string>Application</string>
</property>
<layout class="QGridLayout" name="gridLayout_3">
<item row="0" column="0">
<widget class="QLabel" name="targetSDKLabel">
<property name="text">
<string>Android build SDK:</string>
</property>
</widget>
</item>
<item row="0" column="1">
<widget class="QComboBox" name="targetSDKComboBox"/>
</item>
</layout>
</widget>
</item>
<item row="2" column="0" colspan="2">
<widget class="QGroupBox" name="advancedActions">
<property name="title">
<string>Advanced Actions</string>
</property>
<layout class="QGridLayout" name="gridLayout">
<item row="0" column="0" colspan="2">
<widget class="QCheckBox" name="openPackageLocationCheckBox">
<property name="text">
<string>Open package location after build</string>
</property>
</widget>
</item>
<item row="2" column="0" colspan="2">
<widget class="QCheckBox" name="addDebuggerCheckBox">
<property name="enabled">
<bool>false</bool>
</property>
<property name="toolTip">
<string>Packages debug server with the APK to enable debugging. For the signed APK this option is unchecked by default.</string>
</property>
<property name="text">
<string>Add debug server</string>
</property>
</widget>
</item>
<item row="1" column="0" colspan="2">
<widget class="QCheckBox" name="verboseOutputCheckBox">
<property name="text">
<string>Verbose output</string>
</property>
</widget>
</item>
<item row="3" column="0" colspan="2">
<widget class="QCheckBox" name="ministroOption">
<property name="toolTip">
<string>Uses the external Ministro application to download and maintain Qt libraries.</string>
</property>
<property name="text">
<string>Use Ministro service to install Qt</string>
</property>
</widget>
</item>
</layout>
</widget>
</item>
</layout>
</widget>
<customwidgets>
<customwidget>
<class>Utils::PathChooser</class>
<extends>QWidget</extends>
<header location="global">utils/pathchooser.h</header>
<container>1</container>
</customwidget>
</customwidgets>
<resources/>
<connections>
<connection>
<sender>signPackageCheckBox</sender>
<signal>clicked(bool)</signal>
<receiver>addDebuggerCheckBox</receiver>
<slot>setEnabled(bool)</slot>
<hints>
<hint type="sourcelabel">
<x>113</x>
<y>178</y>
</hint>
<hint type="destinationlabel">
<x>510</x>
<y>452</y>
</hint>
</hints>
</connection>
</connections>
</ui>