Commit 64c17eab authored by Daniel Teske's avatar Daniel Teske

Android: Qt 5.2 deployment add a button to create a Android manifest

The AndroidManifest.xml is no longer needed in as many use cases as
before. So the new deployment does not automatically create a
android manifest. Offer a button on the deployment page to create
a android manifest. This also edits the .pro file adding the
ANDROID_PACKAGE_SOURCE_DIR.

Change-Id: I2655dd6c96e2087732b4d7240b31fe9fcf168600
Reviewed-by: default avatarEskil Abrahamsen Blomfeldt <eskil.abrahamsen-blomfeldt@digia.com>
parent 166a0dff
...@@ -45,7 +45,8 @@ HEADERS += \ ...@@ -45,7 +45,8 @@ HEADERS += \
androiddevicedialog.h \ androiddevicedialog.h \
androiddeployqtstep.h \ androiddeployqtstep.h \
certificatesmodel.h \ certificatesmodel.h \
androiddeployqtwidget.h androiddeployqtwidget.h \
createandroidmanifestwizard.h
SOURCES += \ SOURCES += \
androidconfigurations.cpp \ androidconfigurations.cpp \
...@@ -85,7 +86,8 @@ SOURCES += \ ...@@ -85,7 +86,8 @@ SOURCES += \
androiddevicedialog.cpp \ androiddevicedialog.cpp \
androiddeployqtstep.cpp \ androiddeployqtstep.cpp \
certificatesmodel.cpp \ certificatesmodel.cpp \
androiddeployqtwidget.cpp androiddeployqtwidget.cpp \
createandroidmanifestwizard.cpp
FORMS += \ FORMS += \
androidsettingswidget.ui \ androidsettingswidget.ui \
......
...@@ -34,6 +34,7 @@ ...@@ -34,6 +34,7 @@
#include "androidcreatekeystorecertificate.h" #include "androidcreatekeystorecertificate.h"
#include "androiddeployqtstep.h" #include "androiddeployqtstep.h"
#include "androidmanager.h" #include "androidmanager.h"
#include "createandroidmanifestwizard.h"
#include <projectexplorer/target.h> #include <projectexplorer/target.h>
#include <qt4projectmanager/qt4buildconfiguration.h> #include <qt4projectmanager/qt4buildconfiguration.h>
...@@ -129,6 +130,9 @@ AndroidDeployQtWidget::AndroidDeployQtWidget(AndroidDeployQtStep *step) ...@@ -129,6 +130,9 @@ AndroidDeployQtWidget::AndroidDeployQtWidget(AndroidDeployQtStep *step)
updateInputFileUi(); updateInputFileUi();
connect(m_step, SIGNAL(inputFileChanged()), connect(m_step, SIGNAL(inputFileChanged()),
this, SLOT(updateInputFileUi())); this, SLOT(updateInputFileUi()));
connect(m_ui->createAndroidManifestButton, SIGNAL(clicked()),
this, SLOT(createManifestButton()));
} }
AndroidDeployQtWidget::~AndroidDeployQtWidget() AndroidDeployQtWidget::~AndroidDeployQtWidget()
...@@ -136,6 +140,12 @@ AndroidDeployQtWidget::~AndroidDeployQtWidget() ...@@ -136,6 +140,12 @@ AndroidDeployQtWidget::~AndroidDeployQtWidget()
delete m_ui; delete m_ui;
} }
void AndroidDeployQtWidget::createManifestButton()
{
CreateAndroidManifestWizard wizard(m_step->target());
wizard.exec();
}
void AndroidDeployQtWidget::updateInputFileUi() void AndroidDeployQtWidget::updateInputFileUi()
{ {
Qt4ProjectManager::Qt4Project *project Qt4ProjectManager::Qt4Project *project
......
...@@ -70,6 +70,7 @@ private slots: ...@@ -70,6 +70,7 @@ private slots:
void signPackageCheckBoxToggled(bool checked); void signPackageCheckBoxToggled(bool checked);
void updateInputFileUi(); void updateInputFileUi();
void inputFileComboBoxIndexChanged(); void inputFileComboBoxIndexChanged();
void createManifestButton();
private: private:
virtual QString summaryText() const; virtual QString summaryText() const;
virtual QString displayName() const; virtual QString displayName() const;
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>682</width> <width>682</width>
<height>415</height> <height>467</height>
</rect> </rect>
</property> </property>
<property name="windowTitle"> <property name="windowTitle">
...@@ -177,6 +177,19 @@ The APK will not be usable on any other device.</string> ...@@ -177,6 +177,19 @@ The APK will not be usable on any other device.</string>
</property> </property>
</widget> </widget>
</item> </item>
<item>
<spacer name="verticalSpacer">
<property name="orientation">
<enum>Qt::Vertical</enum>
</property>
<property name="sizeHint" stdset="0">
<size>
<width>20</width>
<height>40</height>
</size>
</property>
</spacer>
</item>
</layout> </layout>
</widget> </widget>
</item> </item>
...@@ -186,21 +199,21 @@ The APK will not be usable on any other device.</string> ...@@ -186,21 +199,21 @@ The APK will not be usable on any other device.</string>
<string>Advanced Actions</string> <string>Advanced Actions</string>
</property> </property>
<layout class="QGridLayout" name="gridLayout"> <layout class="QGridLayout" name="gridLayout">
<item row="3" column="0"> <item row="4" column="0">
<widget class="QPushButton" name="cleanLibsPushButton"> <widget class="QPushButton" name="cleanLibsPushButton">
<property name="text"> <property name="text">
<string>Clean Temporary Libraries Directory on Device</string> <string>Clean Temporary Libraries Directory on Device</string>
</property> </property>
</widget> </widget>
</item> </item>
<item row="4" column="0"> <item row="5" column="0">
<widget class="QPushButton" name="installMinistroButton"> <widget class="QPushButton" name="installMinistroButton">
<property name="text"> <property name="text">
<string>Install Ministro from APK</string> <string>Install Ministro from APK</string>
</property> </property>
</widget> </widget>
</item> </item>
<item row="2" column="0"> <item row="3" column="0">
<widget class="QPushButton" name="resetDefaultDevices"> <widget class="QPushButton" name="resetDefaultDevices">
<property name="text"> <property name="text">
<string>Reset Default Devices</string> <string>Reset Default Devices</string>
...@@ -221,6 +234,13 @@ The APK will not be usable on any other device.</string> ...@@ -221,6 +234,13 @@ The APK will not be usable on any other device.</string>
</property> </property>
</widget> </widget>
</item> </item>
<item row="2" column="0">
<widget class="QPushButton" name="createAndroidManifestButton">
<property name="text">
<string>Create AndroidManifest.xml</string>
</property>
</widget>
</item>
</layout> </layout>
</widget> </widget>
</item> </item>
......
/****************************************************************************
**
** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Digia. For licensing terms and
** conditions see http://qt.digia.com/licensing. For further information
** use the contact form at http://qt.digia.com/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Digia gives you certain additional
** rights. These rights are described in the Digia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#include "createandroidmanifestwizard.h"
#include <projectexplorer/target.h>
#include <qt4projectmanager/qt4project.h>
#include <qt4projectmanager/qt4nodes.h>
#include <proparser/prowriter.h>
#include <QComboBox>
#include <QFormLayout>
#include <QLabel>
#include <QMessageBox>
#include <QVBoxLayout>
#include <qtsupport/qtkitinformation.h>
#include <coreplugin/editormanager/editormanager.h>
using namespace Android;
using namespace Android::Internal;
using Qt4ProjectManager::Qt4Project;
using Qt4ProjectManager::Qt4ProFileNode;
//
// NoApplicationProFilePage
//
NoApplicationProFilePage::NoApplicationProFilePage(CreateAndroidManifestWizard *wizard)
: m_wizard(wizard)
{
QVBoxLayout *layout = new QVBoxLayout(this);
QLabel *label = new QLabel(this);
label->setWordWrap(true);
label->setText(tr("No application .pro file found in this project."));
layout->addWidget(label);
setTitle(tr("No Application .pro File"));
}
//
// ChooseProFilePage
//
ChooseProFilePage::ChooseProFilePage(CreateAndroidManifestWizard *wizard, const QList<Qt4ProFileNode *> &nodes)
: m_wizard(wizard)
{
QFormLayout *fl = new QFormLayout(this);
QLabel *label = new QLabel(this);
label->setWordWrap(true);
label->setText(tr("Select the .pro file for which you want to create a AndroidManifest.xml file"));
fl->addRow(label);
m_comboBox = new QComboBox(this);
foreach (Qt4ProFileNode *node, nodes)
m_comboBox->addItem(node->displayName(), QVariant::fromValue(static_cast<void *>(node))); // TODO something more?
connect(m_comboBox, SIGNAL(currentIndexChanged(int)),
this, SLOT(nodeSelected(int)));
fl->addRow(tr(".pro file:"), m_comboBox);
setTitle(tr("Select a .pro File"));
}
void ChooseProFilePage::nodeSelected(int index)
{
Q_UNUSED(index)
m_wizard->setNode(static_cast<Qt4ProFileNode *>(m_comboBox->currentData().value<void *>()));
}
//
// ChooseDirectoryPage
//
ChooseDirectoryPage::ChooseDirectoryPage(CreateAndroidManifestWizard *wizard)
: m_wizard(wizard), m_androidPackageSourceDir(0)
{
QString androidPackageDir = m_wizard->node()->singleVariableValue(Qt4ProjectManager::AndroidPackageSourceDir);
QFormLayout *fl = new QFormLayout(this);
QLabel *label = new QLabel(this);
label->setWordWrap(true);
fl->addRow(label);
m_androidPackageSourceDir = new Utils::PathChooser(this);
m_androidPackageSourceDir->setExpectedKind(Utils::PathChooser::Directory);
fl->addRow(tr("Android package source directory:"), m_androidPackageSourceDir);
if (androidPackageDir.isEmpty()) {
label->setText(tr("Select the android package source directory. "
"The files in the android package source directory are copied to the builddirectory's "
"android directory and overwrite the default files."));
m_androidPackageSourceDir->setPath(QFileInfo(m_wizard->node()->path()).absolutePath().append(QLatin1String("/android")));
} else {
label->setText(tr("The android manifest file will be created in the ANDROID_PACKAGE_SOURCE_DIR set in the .pro file."));
m_androidPackageSourceDir->setPath(androidPackageDir);
m_androidPackageSourceDir->setReadOnly(true);
}
m_wizard->setDirectory(m_androidPackageSourceDir->path());
connect(m_androidPackageSourceDir, SIGNAL(pathChanged(QString)),
m_wizard, SLOT(setDirectory(QString)));
}
//
// CreateAndroidManifestWizard
//
CreateAndroidManifestWizard::CreateAndroidManifestWizard(ProjectExplorer::Target *target)
: m_target(target), m_node(0)
{
setOption(QWizard::NoBackButtonOnStartPage);
setOption(QWizard::NoCancelButton, false);
setWindowTitle(tr("Create Android Manifest Wizard"));
Qt4Project *project = static_cast<Qt4Project *>(target->project());
QList<Qt4ProFileNode *> nodes = project->applicationProFiles();
if (nodes.isEmpty()) {
// oh uhm can't create anything
addPage(new NoApplicationProFilePage(this));
} else if (nodes.size() == 1) {
setNode(nodes.first());
addPage(new ChooseDirectoryPage(this));
} else {
addPage(new ChooseProFilePage(this, nodes));
addPage(new ChooseDirectoryPage(this));
}
}
Qt4ProjectManager::Qt4ProFileNode *CreateAndroidManifestWizard::node() const
{
return m_node;
}
void CreateAndroidManifestWizard::setNode(Qt4ProjectManager::Qt4ProFileNode *node)
{
m_node = node;
}
void CreateAndroidManifestWizard::setDirectory(const QString &directory)
{
m_directory = directory;
}
QString CreateAndroidManifestWizard::sourceFileName() const
{
QString result;
QtSupport::BaseQtVersion *version = QtSupport::QtKitInformation::qtVersion(m_target->kit());
if (!version)
return result;
Utils::FileName srcPath
= Utils::FileName::fromString(version->qmakeProperty("QT_INSTALL_PREFIX"))
.appendPath(QLatin1String("src/android/java"));
srcPath.appendPath(QLatin1String("AndroidManifest.xml"));
return srcPath.toString();
}
void CreateAndroidManifestWizard::createAndroidManifestFile()
{
if (m_directory.isEmpty())
return;
QDir dir;
if (!QFileInfo(m_directory).exists())
dir.mkpath(m_directory);
QString fileName = m_directory + QLatin1String("/AndroidManifest.xml");
if (QFileInfo(fileName).exists()) {
if (QMessageBox::question(this, tr("Overwrite AndroidManifest.xml"),
tr("Overwrite existing AndroidManifest.xml?"),
QMessageBox::Yes, QMessageBox::No)
== QMessageBox::Yes) {
if (!QFile(m_directory + QLatin1String("/AndroidManifest.xml")).remove()) {
QMessageBox::warning(this, tr("File Removal Error"),
tr("Could not remove file %1.").arg(fileName));
return;
}
} else {
return;
}
}
if (!QFile::copy(sourceFileName(), fileName)) {
QMessageBox::warning(this, tr("File Creation Error"),
tr("Could not create file %1.").arg(fileName));
return;
}
if (m_node->singleVariableValue(Qt4ProjectManager::AndroidPackageSourceDir).isEmpty()) {
// and now time for some magic
QString dir = QFileInfo(fileName).absolutePath();
QString value = QLatin1String("$$PWD/")
+ QDir(m_target->project()->projectDirectory()).relativeFilePath(dir);
bool result =
m_node->setProVariable(QLatin1String("ANDROID_PACKAGE_SOURCE_DIR"), value);
QStringList unChanged;
m_node->addFiles(QStringList(fileName), &unChanged);
if (result == false
|| !unChanged.isEmpty()) {
QMessageBox::warning(this, tr("Project File not Updated"),
tr("Could not update the .pro file %1.").arg(m_node->path()));
}
}
Core::EditorManager::openEditor(fileName);
}
void CreateAndroidManifestWizard::accept()
{
createAndroidManifestFile();
Utils::Wizard::accept();
}
/****************************************************************************
**
** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and Digia. For licensing terms and
** conditions see http://qt.digia.com/licensing. For further information
** use the contact form at http://qt.digia.com/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Digia gives you certain additional
** rights. These rights are described in the Digia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#ifndef CREATEANDROIDMANIFESTWIZARD_H
#define CREATEANDROIDMANIFESTWIZARD_H
#include <utils/pathchooser.h>
#include <utils/wizard.h>
QT_BEGIN_NAMESPACE
class QComboBox;
QT_END_NAMESPACE
namespace ProjectExplorer { class Target; }
namespace Qt4ProjectManager { class Qt4ProFileNode; }
namespace Android {
namespace Internal {
class CreateAndroidManifestWizard;
class NoApplicationProFilePage : public QWizardPage
{
Q_OBJECT
public:
NoApplicationProFilePage(CreateAndroidManifestWizard *wizard);
private:
CreateAndroidManifestWizard *m_wizard;
};
class ChooseProFilePage : public QWizardPage
{
Q_OBJECT
public:
ChooseProFilePage(CreateAndroidManifestWizard *wizard, const QList<Qt4ProjectManager::Qt4ProFileNode *> &nodes);
private slots:
void nodeSelected(int index);
private:
CreateAndroidManifestWizard *m_wizard;
QComboBox *m_comboBox;
};
class ChooseDirectoryPage : public QWizardPage
{
Q_OBJECT
public:
ChooseDirectoryPage(CreateAndroidManifestWizard *wizard);
private:
CreateAndroidManifestWizard *m_wizard;
Utils::PathChooser *m_androidPackageSourceDir;
};
class CreateAndroidManifestWizard : public Utils::Wizard
{
Q_OBJECT
public:
CreateAndroidManifestWizard(ProjectExplorer::Target *target);
Qt4ProjectManager::Qt4ProFileNode *node() const;
void setNode(Qt4ProjectManager::Qt4ProFileNode *node);
QString sourceFileName() const;
void accept();
public slots:
void setDirectory(const QString &directory);
private:
void createAndroidManifestFile();
ProjectExplorer::Target *m_target;
Qt4ProjectManager::Qt4ProFileNode *m_node;
QString m_directory;
};
}
}
#endif // CREATEANDROIDMANIFESTWIZARD_H
...@@ -296,7 +296,7 @@ QStringList Qt4BuildConfiguration::configCommandLineArguments() const ...@@ -296,7 +296,7 @@ QStringList Qt4BuildConfiguration::configCommandLineArguments() const
QStringList result; QStringList result;
BaseQtVersion *version = QtKitInformation::qtVersion(target()->kit()); BaseQtVersion *version = QtKitInformation::qtVersion(target()->kit());
BaseQtVersion::QmakeBuildConfigs defaultBuildConfiguration = BaseQtVersion::QmakeBuildConfigs defaultBuildConfiguration =
version ? version->defaultBuildConfig() : (BaseQtVersion::DebugBuild | BaseQtVersion::BuildAll); version ? version->defaultBuildConfig() : BaseQtVersion::QmakeBuildConfigs(BaseQtVersion::DebugBuild | BaseQtVersion::BuildAll);
BaseQtVersion::QmakeBuildConfigs userBuildConfiguration = m_qmakeBuildConfiguration; BaseQtVersion::QmakeBuildConfigs userBuildConfiguration = m_qmakeBuildConfiguration;
if ((defaultBuildConfiguration & BaseQtVersion::BuildAll) && !(userBuildConfiguration & BaseQtVersion::BuildAll)) if ((defaultBuildConfiguration & BaseQtVersion::BuildAll) && !(userBuildConfiguration & BaseQtVersion::BuildAll))
result << QLatin1String("CONFIG-=debug_and_release"); result << QLatin1String("CONFIG-=debug_and_release");
......
...@@ -1107,45 +1107,37 @@ QStringList Qt4PriFileNode::formResources(const QString &formFile) const ...@@ -1107,45 +1107,37 @@ QStringList Qt4PriFileNode::formResources(const QString &formFile) const
return resourceFiles; return resourceFiles;
} }
void Qt4PriFileNode::changeFiles(const QString &mimeType, bool Qt4PriFileNode::ensureWriteableProFile(const QString &file)
const QStringList &filePaths,
QStringList *notChanged,
ChangeType change)
{ {
if (filePaths.isEmpty())
return;
*notChanged = filePaths;
// Check for modified editors
if (!saveModifiedEditors())
return;
// Ensure that the file is not read only // Ensure that the file is not read only
QFileInfo fi(m_projectFilePath); QFileInfo fi(file);
if (!fi.isWritable()) { if (!fi.isWritable()) {
// Try via vcs manager // Try via vcs manager
Core::IVersionControl *versionControl = Core::VcsManager::findVersionControlForDirectory(fi.absolutePath()); Core::IVersionControl *versionControl = Core::VcsManager::findVersionControlForDirectory(fi.absolutePath());
if (!versionControl || versionControl->vcsOpen(m_projectFilePath)) { if (!versionControl || versionControl->vcsOpen(file)) {
bool makeWritable = QFile::setPermissions(m_projectFilePath, fi.permissions() | QFile::WriteUser); bool makeWritable = QFile::setPermissions(file, fi.permissions() | QFile::WriteUser);
if (!makeWritable) { if (!makeWritable) {
QMessageBox::warning(Core::ICore::mainWindow(), QMessageBox::warning(Core::ICore::mainWindow(),
tr("Failed!"), tr("Failed!"),
tr("Could not write project file %1.").arg(m_projectFilePath)); tr("Could not write project file %1.").arg(file));
return; return false;
} }
} }
} }
return true;
}
QPair<ProFile *, QStringList> Qt4PriFileNode::readProFile(const QString &file)
{
QStringList lines; QStringList lines;
ProFile *includeFile; ProFile *includeFile = 0;
{ {
QString contents; QString contents;
{ {
Utils::FileReader reader; Utils::FileReader reader;
if (!reader.fetch(m_projectFilePath, QIODevice::Text)) { if (!reader.fetch(file, QIODevice::Text)) {