2012-04-18 20:30:57 +03:00
|
|
|
/**************************************************************************
|
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (c) 2014 BogDan Vatra <bog_dan_ro@yahoo.com>
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2012-04-18 20:30:57 +03:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2012-04-18 20:30:57 +03:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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.
|
2012-04-18 20:30:57 +03:00
|
|
|
**
|
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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
|
2012-04-18 20:30:57 +03:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
#include "androidmanager.h"
|
2012-04-18 20:30:57 +03:00
|
|
|
#include "androiddeployconfiguration.h"
|
|
|
|
#include "androidconfigurations.h"
|
|
|
|
#include "androidrunconfiguration.h"
|
|
|
|
#include "androidglobal.h"
|
2013-01-25 16:49:22 +01:00
|
|
|
#include "androidtoolchain.h"
|
2013-09-17 18:24:57 +02:00
|
|
|
#include "androiddeployqtstep.h"
|
2014-06-24 16:47:38 +02:00
|
|
|
#include "androidqtsupport.h"
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2013-04-16 17:27:06 +02:00
|
|
|
#include <coreplugin/documentmanager.h>
|
2013-09-17 18:24:57 +02:00
|
|
|
#include <coreplugin/messagemanager.h>
|
2014-03-11 18:09:23 +01:00
|
|
|
#include <coreplugin/icore.h>
|
2014-06-24 16:47:38 +02:00
|
|
|
|
|
|
|
#include <extensionsystem/pluginmanager.h>
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
#include <projectexplorer/projectexplorer.h>
|
|
|
|
#include <projectexplorer/session.h>
|
|
|
|
#include <projectexplorer/target.h>
|
2013-10-29 16:19:24 +01:00
|
|
|
#include <qmakeprojectmanager/qmakenodes.h>
|
|
|
|
#include <qmakeprojectmanager/qmakeproject.h>
|
|
|
|
#include <qmakeprojectmanager/qmakeprojectmanagerconstants.h>
|
|
|
|
#include <qmakeprojectmanager/qmakebuildconfiguration.h>
|
2012-05-16 16:24:16 +02:00
|
|
|
#include <qtsupport/customexecutablerunconfiguration.h>
|
2012-09-03 18:31:44 +02:00
|
|
|
#include <qtsupport/qtkitinformation.h>
|
2012-04-24 15:49:09 +02:00
|
|
|
#include <qtsupport/qtsupportconstants.h>
|
2014-06-16 18:25:52 +04:00
|
|
|
#include <utils/algorithm.h>
|
2012-04-18 20:30:57 +03:00
|
|
|
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFileSystemWatcher>
|
|
|
|
#include <QList>
|
|
|
|
#include <QProcess>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QApplication>
|
|
|
|
#include <QDomDocument>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
const QLatin1String AndroidDirName("android");
|
|
|
|
const QLatin1String AndroidManifestName("AndroidManifest.xml");
|
|
|
|
const QLatin1String AndroidLibsFileName("/res/values/libs.xml");
|
|
|
|
const QLatin1String AndroidStringsFileName("/res/values/strings.xml");
|
|
|
|
const QLatin1String AndroidDefaultPropertiesName("project.properties");
|
|
|
|
} // anonymous namespace
|
|
|
|
|
|
|
|
namespace Android {
|
|
|
|
namespace Internal {
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::supportsAndroid(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2013-10-29 14:22:31 +01:00
|
|
|
if (!qobject_cast<QmakeProjectManager::QmakeProject *>(target->project()))
|
2012-04-24 15:49:09 +02:00
|
|
|
return false;
|
2012-09-03 18:31:44 +02:00
|
|
|
QtSupport::BaseQtVersion *version = QtSupport::QtKitInformation::qtVersion(target->kit());
|
2012-07-16 11:04:42 +02:00
|
|
|
return version && version->platformName() == QLatin1String(QtSupport::Constants::ANDROID_PLATFORM);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QString AndroidManager::packageName(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
QDomDocument doc;
|
|
|
|
if (!openManifest(target, doc))
|
|
|
|
return QString();
|
|
|
|
QDomElement manifestElem = doc.documentElement();
|
|
|
|
return manifestElem.attribute(QLatin1String("package"));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QString AndroidManager::intentName(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return packageName(target) + QLatin1Char('/') + activityName(target);
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QString AndroidManager::activityName(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
QDomDocument doc;
|
|
|
|
if (!openManifest(target, doc))
|
|
|
|
return QString();
|
|
|
|
QDomElement activityElem = doc.documentElement().firstChildElement(QLatin1String("application")).firstChildElement(QLatin1String("activity"));
|
|
|
|
return activityElem.attribute(QLatin1String("android:name"));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2013-08-28 15:23:04 +02:00
|
|
|
int AndroidManager::minimumSDK(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
QDomDocument doc;
|
|
|
|
if (!openManifest(target, doc))
|
|
|
|
return 0;
|
|
|
|
QDomElement manifestElem = doc.documentElement();
|
|
|
|
QDomElement usesSdk = manifestElem.firstChildElement(QLatin1String("uses-sdk"));
|
|
|
|
if (usesSdk.isNull())
|
|
|
|
return 0;
|
|
|
|
if (usesSdk.hasAttribute(QLatin1String("android:minSdkVersion"))) {
|
|
|
|
bool ok;
|
|
|
|
int tmp = usesSdk.attribute(QLatin1String("android:minSdkVersion")).toInt(&ok);
|
|
|
|
if (ok)
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-08-28 15:20:54 +02:00
|
|
|
QString AndroidManager::buildTargetSDK(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2014-05-07 15:03:43 +03:00
|
|
|
if (!target->activeDeployConfiguration())
|
2013-09-17 18:24:57 +02:00
|
|
|
return QLatin1String("android-9");
|
2014-05-07 15:03:43 +03:00
|
|
|
AndroidDeployQtStep *step = AndroidGlobal::buildStep<AndroidDeployQtStep>(target->activeDeployConfiguration());
|
|
|
|
if (step)
|
|
|
|
return step->buildTargetSdk();
|
|
|
|
return QLatin1String("android-9");
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2013-05-03 15:10:21 +02:00
|
|
|
QString AndroidManager::targetArch(ProjectExplorer::Target *target)
|
|
|
|
{
|
2013-10-29 14:22:31 +01:00
|
|
|
QmakeProjectManager::QmakeProject *pro = qobject_cast<QmakeProjectManager::QmakeProject *>(target->project());
|
2013-05-03 15:10:21 +02:00
|
|
|
if (!pro)
|
|
|
|
return QString();
|
2013-10-29 14:22:31 +01:00
|
|
|
QmakeProjectManager::QmakeProFileNode *node = pro->rootQmakeProjectNode();
|
2013-05-03 15:10:21 +02:00
|
|
|
if (!node)
|
|
|
|
return QString();
|
2013-10-16 11:02:37 +02:00
|
|
|
return node->singleVariableValue(QmakeProjectManager::AndroidArchVar);
|
2013-05-03 15:10:21 +02:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::dirPath(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2014-05-07 15:03:43 +03:00
|
|
|
return target->activeBuildConfiguration()->buildDirectory().appendPath(QLatin1String(Constants::ANDROID_BUILDDIRECTORY));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::manifestPath(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return dirPath(target).appendPath(AndroidManifestName);
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::libsPath(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return dirPath(target).appendPath(AndroidLibsFileName);
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::defaultPropertiesPath(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return dirPath(target).appendPath(AndroidDefaultPropertiesName);
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::apkPath(ProjectExplorer::Target *target, BuildType buildType)
|
|
|
|
{
|
2013-09-17 18:24:57 +02:00
|
|
|
QString packageName = QLatin1String("QtApp");
|
|
|
|
QString buildTypeName;
|
|
|
|
if (buildType == DebugBuild)
|
|
|
|
buildTypeName = QLatin1String("debug");
|
|
|
|
else if (buildType == ReleaseBuildUnsigned)
|
|
|
|
buildTypeName =QLatin1String("release-unsigned");
|
|
|
|
else
|
|
|
|
buildTypeName = QLatin1String("release");
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
return dirPath(target)
|
|
|
|
.appendPath(QLatin1String("bin"))
|
|
|
|
.appendPath(QString::fromLatin1("%1-%2.apk")
|
2013-09-17 18:24:57 +02:00
|
|
|
.arg(packageName)
|
|
|
|
.arg(buildTypeName));
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList AndroidManager::availableTargetApplications(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
QStringList apps;
|
2013-10-29 17:37:39 +01:00
|
|
|
QmakeProjectManager::QmakeProject *qmakeProject = qobject_cast<QmakeProjectManager::QmakeProject *>(target->project());
|
|
|
|
if (!qmakeProject)
|
2013-08-28 16:28:47 +02:00
|
|
|
return apps;
|
2013-10-29 17:37:39 +01:00
|
|
|
foreach (QmakeProjectManager::QmakeProFileNode *proFile, qmakeProject->applicationProFiles()) {
|
2013-10-16 11:02:37 +02:00
|
|
|
if (proFile->projectType() == QmakeProjectManager::ApplicationTemplate) {
|
2012-04-24 15:49:09 +02:00
|
|
|
if (proFile->targetInformation().target.startsWith(QLatin1String("lib"))
|
|
|
|
&& proFile->targetInformation().target.endsWith(QLatin1String(".so")))
|
|
|
|
apps << proFile->targetInformation().target.mid(3, proFile->targetInformation().target.lastIndexOf(QLatin1Char('.')) - 3);
|
|
|
|
else
|
|
|
|
apps << proFile->targetInformation().target;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
apps.sort();
|
|
|
|
return apps;
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2013-04-19 12:27:58 +02:00
|
|
|
bool AndroidManager::bundleQt(ProjectExplorer::Target *target)
|
|
|
|
{
|
2013-09-17 18:24:57 +02:00
|
|
|
AndroidDeployQtStep *androidDeployQtStep
|
|
|
|
= AndroidGlobal::buildStep<AndroidDeployQtStep>(target->activeDeployConfiguration());
|
2013-11-11 22:20:47 +02:00
|
|
|
if (androidDeployQtStep)
|
2013-09-17 18:24:57 +02:00
|
|
|
return androidDeployQtStep->deployAction() == AndroidDeployQtStep::BundleLibrariesDeployment;
|
2013-04-19 12:27:58 +02:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-09-17 18:24:57 +02:00
|
|
|
bool AndroidManager::useLocalLibs(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
AndroidDeployQtStep *androidDeployQtStep
|
|
|
|
= AndroidGlobal::buildStep<AndroidDeployQtStep>(target->activeDeployConfiguration());
|
|
|
|
if (androidDeployQtStep) {
|
|
|
|
return androidDeployQtStep->deployAction() == AndroidDeployQtStep::DebugDeployment
|
|
|
|
|| androidDeployQtStep->deployAction() == AndroidDeployQtStep::BundleLibrariesDeployment;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString AndroidManager::deviceSerialNumber(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
AndroidDeployQtStep *androidDeployQtStep
|
|
|
|
= AndroidGlobal::buildStep<AndroidDeployQtStep>(target->activeDeployConfiguration());
|
|
|
|
if (androidDeployQtStep)
|
|
|
|
return androidDeployQtStep->deviceSerialNumber();
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
Utils::FileName AndroidManager::localLibsRulesFilePath(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-09-03 18:31:44 +02:00
|
|
|
QtSupport::BaseQtVersion *version = QtSupport::QtKitInformation::qtVersion(target->kit());
|
2012-04-24 15:49:09 +02:00
|
|
|
if (!version)
|
|
|
|
return Utils::FileName();
|
2013-02-15 16:07:54 +01:00
|
|
|
return Utils::FileName::fromString(version->qmakeProperty("QT_INSTALL_LIBS"));
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QString AndroidManager::loadLocalLibs(ProjectExplorer::Target *target, int apiLevel)
|
|
|
|
{
|
|
|
|
return loadLocal(target, apiLevel, Lib);
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QString AndroidManager::loadLocalJars(ProjectExplorer::Target *target, int apiLevel)
|
|
|
|
{
|
2013-04-19 12:27:58 +02:00
|
|
|
ItemType type = bundleQt(target) ? BundledJar : Jar;
|
|
|
|
return loadLocal(target, apiLevel, type);
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2013-01-06 16:23:08 +02:00
|
|
|
QString AndroidManager::loadLocalJarsInitClasses(ProjectExplorer::Target *target, int apiLevel)
|
|
|
|
{
|
2013-04-19 12:27:58 +02:00
|
|
|
ItemType type = bundleQt(target) ? BundledJar : Jar;
|
|
|
|
return loadLocal(target, apiLevel, type, QLatin1String("initClass"));
|
2013-01-06 16:23:08 +02:00
|
|
|
}
|
|
|
|
|
2014-05-07 15:03:43 +03:00
|
|
|
QPair<int, int> AndroidManager::apiLevelRange()
|
2013-08-14 13:39:45 +02:00
|
|
|
{
|
2014-05-07 15:03:43 +03:00
|
|
|
return qMakePair(9, 20);
|
2013-08-14 13:39:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QString AndroidManager::androidNameForApiLevel(int x)
|
|
|
|
{
|
|
|
|
switch (x) {
|
|
|
|
case 4:
|
|
|
|
return QLatin1String("Android 1.6");
|
|
|
|
case 5:
|
|
|
|
return QLatin1String("Android 2.0");
|
|
|
|
case 6:
|
|
|
|
return QLatin1String("Android 2.0.1");
|
|
|
|
case 7:
|
|
|
|
return QLatin1String("Android 2.1.x");
|
|
|
|
case 8:
|
|
|
|
return QLatin1String("Android 2.2.x");
|
|
|
|
case 9:
|
|
|
|
return QLatin1String("Android 2.3, 2.3.1, 2.3.2");
|
|
|
|
case 10:
|
|
|
|
return QLatin1String("Android 2.3.3, 2.3.4");
|
|
|
|
case 11:
|
|
|
|
return QLatin1String("Android 3.0.x");
|
|
|
|
case 12:
|
|
|
|
return QLatin1String("Android 3.1.x");
|
|
|
|
case 13:
|
|
|
|
return QLatin1String("Android 3.2");
|
|
|
|
case 14:
|
|
|
|
return QLatin1String("Android 4.0, 4.0.1, 4.0.2");
|
|
|
|
case 15:
|
|
|
|
return QLatin1String("Android 4.0.3, 4.0.4");
|
|
|
|
case 16:
|
|
|
|
return QLatin1String("Android 4.1, 4.1.1");
|
|
|
|
case 17:
|
|
|
|
return QLatin1String("Android 4.2, 4.2.2");
|
|
|
|
case 18:
|
|
|
|
return QLatin1String("Android 4.3");
|
2013-11-12 12:14:33 +01:00
|
|
|
case 19:
|
|
|
|
return QLatin1String("Android 4.4");
|
2014-07-16 17:39:24 +02:00
|
|
|
case 20:
|
|
|
|
return QLatin1String("Android L"); // prelimary name?
|
2013-08-14 13:39:45 +02:00
|
|
|
default:
|
2014-07-16 17:36:21 +02:00
|
|
|
return tr("Unknown Android version. API Level: %1").arg(QString::number(x));
|
2013-08-14 13:39:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList AndroidManager::qtLibs(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return libsXml(target, QLatin1String("qt_libs"));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList AndroidManager::prebundledLibs(ProjectExplorer::Target *target)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
return libsXml(target, QLatin1String("bundled_libs"));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::openLibsXml(ProjectExplorer::Target *target, QDomDocument &doc)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-11-06 16:22:53 +01:00
|
|
|
return openXmlFile(doc, libsPath(target));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::saveLibsXml(ProjectExplorer::Target *target, QDomDocument &doc)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2014-05-07 15:03:43 +03:00
|
|
|
return saveXmlFile(doc, libsPath(target));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
void AndroidManager::raiseError(const QString &reason)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2013-10-09 16:37:42 +02:00
|
|
|
QMessageBox::critical(0, tr("Error creating Android templates."), reason);
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2013-01-06 16:23:08 +02:00
|
|
|
QString AndroidManager::loadLocal(ProjectExplorer::Target *target, int apiLevel, ItemType item, const QString &attribute)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
QString itemType;
|
|
|
|
if (item == Lib)
|
|
|
|
itemType = QLatin1String("lib");
|
2013-04-19 12:27:58 +02:00
|
|
|
else if (item == BundledFile)
|
|
|
|
itemType = QLatin1String("bundled");
|
|
|
|
else // Jar or BundledJar
|
2012-04-24 15:49:09 +02:00
|
|
|
itemType = QLatin1String("jar");
|
|
|
|
|
|
|
|
QString localLibs;
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2013-02-15 16:07:54 +01:00
|
|
|
QDir rulesFilesDir(localLibsRulesFilePath(target).toString());
|
|
|
|
if (!rulesFilesDir.exists())
|
2012-04-24 15:49:09 +02:00
|
|
|
return localLibs;
|
|
|
|
|
|
|
|
QStringList libs;
|
|
|
|
libs << qtLibs(target) << prebundledLibs(target);
|
2013-02-15 16:07:54 +01:00
|
|
|
|
|
|
|
QFileInfoList rulesFiles = rulesFilesDir.entryInfoList(QStringList() << QLatin1String("*.xml"),
|
|
|
|
QDir::Files | QDir::Readable);
|
|
|
|
|
|
|
|
QStringList dependencyLibs;
|
|
|
|
QStringList replacedLibs;
|
|
|
|
foreach (QFileInfo rulesFile, rulesFiles) {
|
|
|
|
if (rulesFile.baseName() != QLatin1String("rules")
|
|
|
|
&& !rulesFile.baseName().endsWith(QLatin1String("-android-dependencies"))) {
|
|
|
|
continue;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2013-02-15 16:07:54 +01:00
|
|
|
QDomDocument doc;
|
|
|
|
if (!openXmlFile(doc, Utils::FileName::fromString(rulesFile.absoluteFilePath())))
|
|
|
|
return localLibs;
|
|
|
|
|
|
|
|
QDomElement element = doc.documentElement().firstChildElement(QLatin1String("platforms")).firstChildElement(itemType + QLatin1Char('s')).firstChildElement(QLatin1String("version"));
|
|
|
|
while (!element.isNull()) {
|
|
|
|
if (element.attribute(QLatin1String("value")).toInt() == apiLevel) {
|
|
|
|
if (element.hasAttribute(QLatin1String("symlink")))
|
|
|
|
apiLevel = element.attribute(QLatin1String("symlink")).toInt();
|
|
|
|
break;
|
2012-04-24 15:49:09 +02:00
|
|
|
}
|
2013-02-15 16:07:54 +01:00
|
|
|
element = element.nextSiblingElement(QLatin1String("version"));
|
|
|
|
}
|
|
|
|
|
|
|
|
element = doc.documentElement().firstChildElement(QLatin1String("dependencies")).firstChildElement(QLatin1String("lib"));
|
|
|
|
while (!element.isNull()) {
|
|
|
|
if (libs.contains(element.attribute(QLatin1String("name")))) {
|
|
|
|
QDomElement libElement = element.firstChildElement(QLatin1String("depends")).firstChildElement(itemType);
|
|
|
|
while (!libElement.isNull()) {
|
2013-04-19 12:27:58 +02:00
|
|
|
if (libElement.attribute(QLatin1String("bundling")).toInt() == (item == BundledJar ? 1 : 0)) {
|
|
|
|
if (libElement.hasAttribute(attribute)) {
|
2013-09-06 15:14:11 +02:00
|
|
|
QString dependencyLib = libElement.attribute(attribute);
|
|
|
|
if (dependencyLib.contains(QLatin1String("%1")))
|
|
|
|
dependencyLib = dependencyLib.arg(apiLevel);
|
2013-05-16 03:58:26 +02:00
|
|
|
if (libElement.hasAttribute(QLatin1String("extends"))) {
|
|
|
|
const QString extends = libElement.attribute(QLatin1String("extends"));
|
2013-11-11 22:20:47 +02:00
|
|
|
if (libs.contains(extends))
|
2013-05-16 03:58:26 +02:00
|
|
|
dependencyLibs << dependencyLib;
|
|
|
|
} else if (!dependencyLibs.contains(dependencyLib)) {
|
2013-04-19 12:27:58 +02:00
|
|
|
dependencyLibs << dependencyLib;
|
2013-05-16 03:58:26 +02:00
|
|
|
}
|
2013-04-19 12:27:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (libElement.hasAttribute(QLatin1String("replaces"))) {
|
2013-09-06 15:14:11 +02:00
|
|
|
QString replacedLib = libElement.attribute(QLatin1String("replaces"));
|
|
|
|
if (replacedLib.contains(QLatin1String("%1")))
|
|
|
|
replacedLib = replacedLib.arg(apiLevel);
|
2013-04-19 12:27:58 +02:00
|
|
|
if (!replacedLibs.contains(replacedLib))
|
|
|
|
replacedLibs << replacedLib;
|
|
|
|
}
|
2013-02-15 16:07:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
libElement = libElement.nextSiblingElement(itemType);
|
|
|
|
}
|
|
|
|
|
|
|
|
libElement = element.firstChildElement(QLatin1String("replaces")).firstChildElement(itemType);
|
|
|
|
while (!libElement.isNull()) {
|
|
|
|
if (libElement.hasAttribute(attribute)) {
|
|
|
|
QString replacedLib = libElement.attribute(attribute).arg(apiLevel);
|
|
|
|
if (!replacedLibs.contains(replacedLib))
|
|
|
|
replacedLibs << replacedLib;
|
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
|
2013-02-15 16:07:54 +01:00
|
|
|
libElement = libElement.nextSiblingElement(itemType);
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
2013-02-15 16:07:54 +01:00
|
|
|
element = element.nextSiblingElement(QLatin1String("lib"));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
}
|
2013-02-15 16:07:54 +01:00
|
|
|
|
|
|
|
// The next loop requires all library names to end with a ":" so we append one
|
|
|
|
// to the end after joining.
|
2013-02-18 16:20:28 +01:00
|
|
|
localLibs = dependencyLibs.join(QLatin1String(":")) + QLatin1Char(':');
|
2013-02-15 16:07:54 +01:00
|
|
|
foreach (QString replacedLib, replacedLibs)
|
|
|
|
localLibs.remove(replacedLib + QLatin1Char(':'));
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
return localLibs;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-11-06 16:22:53 +01:00
|
|
|
bool AndroidManager::openXmlFile(QDomDocument &doc, const Utils::FileName &fileName)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
QFile f(fileName.toString());
|
|
|
|
if (!f.open(QIODevice::ReadOnly))
|
2012-04-18 20:30:57 +03:00
|
|
|
return false;
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
if (!doc.setContent(f.readAll())) {
|
2014-04-17 14:09:47 +02:00
|
|
|
raiseError(tr("Cannot parse \"%1\".").arg(fileName.toUserOutput()));
|
2012-04-18 20:30:57 +03:00
|
|
|
return false;
|
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
return true;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2014-05-07 15:03:43 +03:00
|
|
|
bool AndroidManager::saveXmlFile(QDomDocument &doc, const Utils::FileName &fileName)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
QFile f(fileName.toString());
|
|
|
|
if (!f.open(QIODevice::WriteOnly)) {
|
2014-04-17 14:09:47 +02:00
|
|
|
raiseError(tr("Cannot open \"%1\".").arg(fileName.toUserOutput()));
|
2012-04-24 15:49:09 +02:00
|
|
|
return false;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
return f.write(doc.toByteArray(4)) >= 0;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::openManifest(ProjectExplorer::Target *target, QDomDocument &doc)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-11-06 16:22:53 +01:00
|
|
|
return openXmlFile(doc, manifestPath(target));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::saveManifest(ProjectExplorer::Target *target, QDomDocument &doc)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2013-04-16 17:27:06 +02:00
|
|
|
Core::FileChangeBlocker blocker(manifestPath(target).toString());
|
2014-05-07 15:03:43 +03:00
|
|
|
return saveXmlFile(doc, manifestPath(target));
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList AndroidManager::libsXml(ProjectExplorer::Target *target, const QString &tag)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
|
|
|
QStringList libs;
|
|
|
|
QDomDocument doc;
|
2012-04-24 15:49:09 +02:00
|
|
|
if (!openLibsXml(target, doc))
|
2012-04-18 20:30:57 +03:00
|
|
|
return libs;
|
|
|
|
QDomElement arrayElem = doc.documentElement().firstChildElement(QLatin1String("array"));
|
|
|
|
while (!arrayElem.isNull()) {
|
|
|
|
if (arrayElem.attribute(QLatin1String("name")) == tag) {
|
|
|
|
arrayElem = arrayElem.firstChildElement(QLatin1String("item"));
|
|
|
|
while (!arrayElem.isNull()) {
|
|
|
|
libs << arrayElem.text();
|
|
|
|
arrayElem = arrayElem.nextSiblingElement(QLatin1String("item"));
|
|
|
|
}
|
|
|
|
return libs;
|
|
|
|
}
|
|
|
|
arrayElem = arrayElem.nextSiblingElement(QLatin1String("array"));
|
|
|
|
}
|
|
|
|
return libs;
|
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
bool AndroidManager::setLibsXml(ProjectExplorer::Target *target, const QStringList &libs, const QString &tag)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
|
|
|
QDomDocument doc;
|
2012-04-24 15:49:09 +02:00
|
|
|
if (!openLibsXml(target, doc))
|
2012-04-18 20:30:57 +03:00
|
|
|
return false;
|
|
|
|
QDomElement arrayElem = doc.documentElement().firstChildElement(QLatin1String("array"));
|
|
|
|
while (!arrayElem.isNull()) {
|
|
|
|
if (arrayElem.attribute(QLatin1String("name")) == tag) {
|
|
|
|
doc.documentElement().removeChild(arrayElem);
|
|
|
|
arrayElem = doc.createElement(QLatin1String("array"));
|
|
|
|
arrayElem.setAttribute(QLatin1String("name"), tag);
|
|
|
|
foreach (const QString &lib, libs) {
|
|
|
|
QDomElement item = doc.createElement(QLatin1String("item"));
|
|
|
|
item.appendChild(doc.createTextNode(lib));
|
|
|
|
arrayElem.appendChild(item);
|
|
|
|
}
|
|
|
|
doc.documentElement().appendChild(arrayElem);
|
2012-04-24 15:49:09 +02:00
|
|
|
return saveLibsXml(target, doc);
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
arrayElem = arrayElem.nextSiblingElement(QLatin1String("array"));
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList AndroidManager::dependencies(const Utils::FileName &readelfPath, const QString &lib)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
QStringList libs;
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QProcess readelfProc;
|
|
|
|
readelfProc.start(readelfPath.toString(), QStringList() << QLatin1String("-d") << QLatin1String("-W") << lib);
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
if (!readelfProc.waitForFinished(-1)) {
|
2012-12-10 23:43:21 +00:00
|
|
|
readelfProc.kill();
|
2012-04-24 15:49:09 +02:00
|
|
|
return libs;
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
QList<QByteArray> lines = readelfProc.readAll().trimmed().split('\n');
|
|
|
|
foreach (const QByteArray &line, lines) {
|
|
|
|
if (line.contains("(NEEDED)") && line.contains("Shared library:") ) {
|
|
|
|
const int pos = line.lastIndexOf('[') + 1;
|
|
|
|
libs << QString::fromLatin1(line.mid(pos, line.lastIndexOf(']') - pos));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return libs;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
int AndroidManager::setLibraryLevel(const QString &library, LibrariesMap &mapLibs)
|
2012-04-18 20:30:57 +03:00
|
|
|
{
|
2012-04-24 15:49:09 +02:00
|
|
|
int maxlevel = mapLibs[library].level;
|
|
|
|
if (maxlevel > 0)
|
|
|
|
return maxlevel;
|
|
|
|
foreach (QString lib, mapLibs[library].dependencies) {
|
|
|
|
foreach (const QString &key, mapLibs.keys()) {
|
|
|
|
if (library == key)
|
|
|
|
continue;
|
|
|
|
if (key == lib) {
|
|
|
|
int libLevel = mapLibs[key].level;
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
if (libLevel < 0)
|
|
|
|
libLevel = setLibraryLevel(key, mapLibs);
|
2012-04-18 20:30:57 +03:00
|
|
|
|
2012-04-24 15:49:09 +02:00
|
|
|
if (libLevel > maxlevel)
|
|
|
|
maxlevel = libLevel;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
2012-04-24 15:49:09 +02:00
|
|
|
if (mapLibs[library].level < 0)
|
|
|
|
mapLibs[library].level = maxlevel + 1;
|
|
|
|
return maxlevel + 1;
|
2012-04-18 20:30:57 +03:00
|
|
|
}
|
|
|
|
|
2013-09-17 18:24:57 +02:00
|
|
|
void AndroidManager::cleanLibsOnDevice(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
const QString targetArch = AndroidManager::targetArch(target);
|
2013-10-09 16:24:25 +02:00
|
|
|
if (targetArch.isEmpty())
|
|
|
|
return;
|
2013-09-17 18:24:57 +02:00
|
|
|
int deviceAPILevel = AndroidManager::minimumSDK(target);
|
2013-12-16 20:19:07 +01:00
|
|
|
AndroidDeviceInfo info = AndroidConfigurations::showDeviceDialog(target->project(), deviceAPILevel, targetArch);
|
2013-09-17 18:24:57 +02:00
|
|
|
if (info.serialNumber.isEmpty()) // aborted
|
|
|
|
return;
|
|
|
|
|
|
|
|
deviceAPILevel = info.sdk;
|
|
|
|
QString deviceSerialNumber = info.serialNumber;
|
|
|
|
|
|
|
|
if (info.type == AndroidDeviceInfo::Emulator) {
|
2013-12-16 20:19:07 +01:00
|
|
|
deviceSerialNumber = AndroidConfigurations::currentConfig().startAVD(deviceSerialNumber, deviceAPILevel, targetArch);
|
2013-09-17 18:24:57 +02:00
|
|
|
if (deviceSerialNumber.isEmpty())
|
2013-10-09 16:37:42 +02:00
|
|
|
Core::MessageManager::write(tr("Starting Android virtual device failed."));
|
2013-09-17 18:24:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QProcess *process = new QProcess();
|
|
|
|
QStringList arguments = AndroidDeviceInfo::adbSelector(deviceSerialNumber);
|
|
|
|
arguments << QLatin1String("shell") << QLatin1String("rm") << QLatin1String("-r") << QLatin1String("/data/local/tmp/qt");
|
|
|
|
process->connect(process, SIGNAL(finished(int)), process, SLOT(deleteLater()));
|
2013-12-16 20:19:07 +01:00
|
|
|
const QString adb = AndroidConfigurations::currentConfig().adbToolPath().toString();
|
2013-09-17 18:24:57 +02:00
|
|
|
Core::MessageManager::write(adb + QLatin1Char(' ') + arguments.join(QLatin1String(" ")));
|
|
|
|
process->start(adb, arguments);
|
|
|
|
if (!process->waitForStarted(500))
|
|
|
|
delete process;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AndroidManager::installQASIPackage(ProjectExplorer::Target *target, const QString &packagePath)
|
|
|
|
{
|
|
|
|
const QString targetArch = AndroidManager::targetArch(target);
|
2013-10-09 16:24:25 +02:00
|
|
|
if (targetArch.isEmpty())
|
|
|
|
return;
|
2013-09-17 18:24:57 +02:00
|
|
|
int deviceAPILevel = AndroidManager::minimumSDK(target);
|
2013-12-16 20:19:07 +01:00
|
|
|
AndroidDeviceInfo info = AndroidConfigurations::showDeviceDialog(target->project(), deviceAPILevel, targetArch);
|
2013-09-17 18:24:57 +02:00
|
|
|
if (info.serialNumber.isEmpty()) // aborted
|
|
|
|
return;
|
|
|
|
|
|
|
|
deviceAPILevel = info.sdk;
|
|
|
|
QString deviceSerialNumber = info.serialNumber;
|
|
|
|
if (info.type == AndroidDeviceInfo::Emulator) {
|
2013-12-16 20:19:07 +01:00
|
|
|
deviceSerialNumber = AndroidConfigurations::currentConfig().startAVD(deviceSerialNumber, deviceAPILevel, targetArch);
|
2013-09-17 18:24:57 +02:00
|
|
|
if (deviceSerialNumber.isEmpty())
|
2013-10-09 16:37:42 +02:00
|
|
|
Core::MessageManager::write(tr("Starting Android virtual device failed."));
|
2013-09-17 18:24:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QProcess *process = new QProcess();
|
|
|
|
QStringList arguments = AndroidDeviceInfo::adbSelector(deviceSerialNumber);
|
|
|
|
arguments << QLatin1String("install") << QLatin1String("-r ") << packagePath;
|
|
|
|
|
|
|
|
process->connect(process, SIGNAL(finished(int)), process, SLOT(deleteLater()));
|
2013-12-16 20:19:07 +01:00
|
|
|
const QString adb = AndroidConfigurations::currentConfig().adbToolPath().toString();
|
2013-09-17 18:24:57 +02:00
|
|
|
Core::MessageManager::write(adb + QLatin1Char(' ') + arguments.join(QLatin1String(" ")));
|
|
|
|
process->start(adb, arguments);
|
|
|
|
if (!process->waitForFinished(500))
|
|
|
|
delete process;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AndroidManager::checkKeystorePassword(const QString &keystorePath, const QString &keystorePasswd)
|
|
|
|
{
|
|
|
|
if (keystorePasswd.isEmpty())
|
|
|
|
return false;
|
|
|
|
QStringList arguments;
|
|
|
|
arguments << QLatin1String("-list")
|
|
|
|
<< QLatin1String("-keystore")
|
|
|
|
<< keystorePath
|
|
|
|
<< QLatin1String("--storepass")
|
|
|
|
<< keystorePasswd;
|
|
|
|
QProcess proc;
|
2013-12-16 20:19:07 +01:00
|
|
|
proc.start(AndroidConfigurations::currentConfig().keytoolPath().toString(), arguments);
|
2013-12-10 12:58:10 +01:00
|
|
|
if (!proc.waitForStarted(4000))
|
2013-09-17 18:24:57 +02:00
|
|
|
return false;
|
2013-12-10 12:58:10 +01:00
|
|
|
if (!proc.waitForFinished(4000)) {
|
2013-09-17 18:24:57 +02:00
|
|
|
proc.kill();
|
|
|
|
proc.waitForFinished();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return proc.exitCode() == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AndroidManager::checkCertificatePassword(const QString &keystorePath, const QString &keystorePasswd, const QString &alias, const QString &certificatePasswd)
|
|
|
|
{
|
|
|
|
// assumes that the keystore password is correct
|
|
|
|
QStringList arguments;
|
|
|
|
arguments << QLatin1String("-certreq")
|
|
|
|
<< QLatin1String("-keystore")
|
|
|
|
<< keystorePath
|
|
|
|
<< QLatin1String("--storepass")
|
|
|
|
<< keystorePasswd
|
|
|
|
<< QLatin1String("-alias")
|
|
|
|
<< alias
|
|
|
|
<< QLatin1String("-keypass");
|
|
|
|
if (certificatePasswd.isEmpty())
|
|
|
|
arguments << keystorePasswd;
|
|
|
|
else
|
|
|
|
arguments << certificatePasswd;
|
|
|
|
|
|
|
|
QProcess proc;
|
2013-12-16 20:19:07 +01:00
|
|
|
proc.start(AndroidConfigurations::currentConfig().keytoolPath().toString(), arguments);
|
2013-12-10 12:58:10 +01:00
|
|
|
if (!proc.waitForStarted(4000))
|
2013-09-17 18:24:57 +02:00
|
|
|
return false;
|
2013-12-10 12:58:10 +01:00
|
|
|
if (!proc.waitForFinished(4000)) {
|
2013-09-17 18:24:57 +02:00
|
|
|
proc.kill();
|
|
|
|
proc.waitForFinished();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return proc.exitCode() == 0;
|
|
|
|
}
|
|
|
|
|
2014-06-24 16:47:10 +02:00
|
|
|
bool AndroidManager::checkForQt51Files(Utils::FileName fileName)
|
2013-09-17 18:24:57 +02:00
|
|
|
{
|
|
|
|
fileName.appendPath(QLatin1String("android")).appendPath(QLatin1String("version.xml"));
|
|
|
|
if (!fileName.toFileInfo().exists())
|
|
|
|
return false;
|
|
|
|
QDomDocument dstVersionDoc;
|
|
|
|
if (!AndroidManager::openXmlFile(dstVersionDoc, fileName))
|
|
|
|
return false;
|
|
|
|
return dstVersionDoc.documentElement().attribute(QLatin1String("value")).toDouble() < 5.2;
|
|
|
|
}
|
|
|
|
|
2014-06-24 16:47:38 +02:00
|
|
|
AndroidQtSupport *AndroidManager::androidQtSupport(ProjectExplorer::Target *target)
|
|
|
|
{
|
|
|
|
QList<AndroidQtSupport *> providerList = ExtensionSystem::PluginManager::getObjects<AndroidQtSupport>();
|
|
|
|
foreach (AndroidQtSupport *provider, providerList) {
|
|
|
|
if (provider->canHandle(target))
|
|
|
|
return provider;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-18 20:30:57 +03:00
|
|
|
} // namespace Internal
|
2013-10-16 11:02:37 +02:00
|
|
|
} // namespace Android
|