如何将qt5 qml插件部署到android?

时间:2013-08-19 08:06:22

标签: android qml qt5

我正在尝试从Qt 5.1.0 android安装中导入TimeExample Qt Quick Extension Plugin。

libqmlqtimeexampleplugin.so已在build-plugins-Android_for_arm_GCC_4_6_Qt_5_1_0-Debug/imports

中成功构建

然后我从Qt Creator创建了简单的Qt Quick2应用程序(内置元素)。我应该在应用程序项目文件中添加什么来获取输出“.apk”包中的QML插件?

现在说:

  

W / Qt(23528):assets:/qml/TimeExampleTest/main.qml:2():assets:/qml/TimeExampleTest/main.qml:2:1:模块“TimeExample”未安装

main.qml

 import QtQuick 2.0

 import TimeExample 1.0 // import types from the plugin

 Rectangle {
    width: 360
    height: 360
    Text {
        text: qsTr("Hello World")
        anchors.centerIn: parent
    }
    MouseArea {
       anchors.fill: parent
        onClicked: {
          Qt.quit();
       }
    }

   Clock { // this class is defined in QML (imports/TimeExample/Clock.qml)

        Time { // this class is defined in C++ (plugin.cpp)
            id: time
      }

        hours: time.hour
      minutes: time.minute

    }
}

TimeExampleTest.pro

folder_01.source = qml/TimeExampleTest

 folder_01.target = qml

 folder_02.source = /home/artem/Projects/Veedo/Test/build-plugins-Android_for_arm_GCC_4_6_Qt_5_1_0-Debug/imports/TimeExample

 folder_02.target = imports

 DEPLOYMENTFOLDERS = folder_01 folder_02

 QML_IMPORT_PATH = /home/artem/Projects/Veedo/Test/build-plugins-Android_for_arm_GCC_4_6_Qt_5_1_0-Debug/imports/TimeExample

 SOURCES += main.cpp

 include(qtquick2applicationviewer/qtquick2applicationviewer.pri)

 qtcAddDeployment()

 OTHER_FILES += \
    android/src/org/kde/necessitas/ministro/IMinistro.aidl \
    android/src/org/kde/necessitas/ministro/IMinistroCallback.aidl \
    android/src/org/qtproject/qt5/android/bindings/QtActivity.java \
    android/src/org/qtproject/qt5/android/bindings/QtApplication.java \
    android/AndroidManifest.xml \
    android/version.xml \
    android/res/values-ja/strings.xml \
    android/res/values-rs/strings.xml \
    android/res/values-zh-rTW/strings.xml \
    android/res/values-fa/strings.xml \
    android/res/values-ru/strings.xml \
    android/res/values-fr/strings.xml \
    android/res/values-ro/strings.xml \
    android/res/values-el/strings.xml \
    android/res/values-ms/strings.xml \
    android/res/values-nb/strings.xml \
    android/res/values-et/strings.xml \
    android/res/values-pl/strings.xml \
    android/res/values-pt-rBR/strings.xml \
    android/res/values-es/strings.xml \
    android/res/values-id/strings.xml \
    android/res/values-de/strings.xml \
    android/res/values-it/strings.xml \
    android/res/values-zh-rCN/strings.xml \
    android/res/values/strings.xml \
    android/res/values/libs.xml \
    android/res/layout/splash.xml \
    android/res/values-nl/strings.xml

3 个答案:

答案 0 :(得分:6)

使用Qt 5.3,我们只通过将插件部署到官方Qt QML模块所在的QT_INSTALL_QML目录,设法通过Qt Android应用程序识别我们的QML插件。在我们的例子中,这个目录是/opt/Qt/5.3/android_armv7/qml。

插件端

我们的插件.pro文件如下:

TEMPLATE = lib
TARGET = prova
QT += qml quick multimedia
CONFIG += qt plugin c++11 console
CONFIG -= android_install
TARGET = $$qtLibraryTarget($$TARGET)
uri = com.mycompany.qmlcomponents

# Input
SOURCES += \
    src1.cpp \
    src2.cpp

HEADERS += \
    src1.h \
    src2.h

##The below is generated automatically by Qt Creator when you create a new "Qt Quick 2 Extension Plugin" project for Android

#Copies the qmldir file to the build directory
!equals(_PRO_FILE_PWD_, $$OUT_PWD) {
    copy_qmldir.target = $$OUT_PWD/qmldir
    copy_qmldir.depends = $$_PRO_FILE_PWD_/qmldir
    copy_qmldir.commands = $(COPY_FILE) \"$$replace(copy_qmldir.depends, /, $$QMAKE_DIR_SEP)\" \"$$replace(copy_qmldir.target, /, $$QMAKE_DIR_SEP)\"
    QMAKE_EXTRA_TARGETS += copy_qmldir
    PRE_TARGETDEPS += $$copy_qmldir.target
}

#Copies the qmldir file and the built plugin .so to the QT_INSTALL_QML directory
qmldir.files = qmldir
unix {
    installPath = $$[QT_INSTALL_QML]/$$replace(uri, \\., /)
    qmldir.path = $$installPath
    target.path = $$installPath
    INSTALLS += target qmldir
}

我们的qmldir(在插件源树根目录中)文件是:

module com.mycompany.qmlcomponents
plugin prova

申请方

.pro文件如下所示:

TEMPLATE = app

QT += qml quick widgets multimedia

CONFIG+= console
SOURCES += main.cpp

RESOURCES += qml.qrc

# Additional import path used to resolve QML modules in Qt Creator's code model
QML_IMPORT_PATH =
# Default rules for deployment.
include(deployment.pri)

contains(ANDROID_TARGET_ARCH,armeabi-v7a) {
    ANDROID_EXTRA_LIBS = \
        /opt/Qt/5.3/android_armv7/qml/com/mycompany/qmlcomponents/libprova.so
}

我们实际上并不知道是否需要包含额外的libprova.so。这很可能不是。

main.cpp看起来像:

#include <QApplication>
#include <QQmlApplicationEngine>
int main(int argc, char *argv[]){
    QApplication app(argc, argv);
    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:///main.qml")));
    return app.exec();
}

main.qml只包含以下插件:

import QtQuick 2.2
import QtQuick.Controls 1.1
import QtMultimedia 5.0

import com.mycompany.qmlcomponents 1.0

...

构建和部署

我们构建和部署插件的方式是qmake(来自Qt的android-armv7工具链),然后是make install。它将qmldir文件和插件.so安装到QT_INSTALL_QML目录中。

我们构建和部署使用该插件的项目的方式是qmake(再次,来自Qt的android-armv7工具链),然后是make install INSTALL_ROOT=.(安装到构建目录),然后运行androiddeployqt。最后一个命令使用资源/中的qmldirs和libs /中的库创建Android项目结构,并通过ant将所有内容捆绑在apk中。有关此过程的详细信息,请参阅http://qt-project.org/wiki/Android

简而言之,我们只能通过将其置于私有Qt qml目录中来获取我们在Android项目中识别的QML插件。

答案 1 :(得分:0)

在阅读了Qt的关于QAbstractFileEngineHandler的消息来源后,我在Qt中发现了一些丑陋的东西:

  1. Qt确实实现了某种机制来提供&#34; searchpath&#34; (不是URL)喜欢&#34; qrc:&#34;,&#34; assets:&#34;,其中内核是QAbstractFileEngineHandler。

  2. QAbstractFileEngineHandler在Qt5中不再公开,只提供QFile,QFileInfo,QDir,QIcon和QFile提供的东西的摘要,包括QImage,以及专门用于Android资产的QAndroidAssetsFileEngineHandler。

  3. 所以SearchPath架构与QUrl不兼容,即使它们看起来如此相似,因为你知道QtQuick是setSource到某个URL,所以qml文件不会被&#34; assets:&#34;直接前缀。

  4. 您可以在Qt的文档中找到[static] void QDir::setSearchPaths(const QString &prefix, const QStringList &searchPaths)下面的内容:

    QDir::setSearchPaths("icons", QStringList(QDir::homePath() + "/images"));
    QDir::setSearchPaths("docs", QStringList(":/embeddedDocuments"));
    
    ...    
    QPixmap pixmap("icons:undo.png"); // will look for undo.png in QDir::homePath() + "/images"    
    QFile file("docs:design.odf"); // will look in the :/embeddedDocuments resource path
    

    您会发现此代码段不起作用,因为QPixmap没有从QAbstractFileEngineHandler加载图像,这可能是Qt的未记录的更改。另一方面,关于searchpath的QtCore单元测试仅在QFile上进行了测试

  5. 此外,void addResourceSearchPath(const QString &path)已废弃,但setSearchPath有缺陷,我对此感到非常困惑。也许我现在应该看看QUrl:)

答案 2 :(得分:0)

这是我的工作方式。

将此添加到您的应用程序代码中:

QQuickView *m_view;
m_view->engine()->addImportPath("assets:/");
m_view->engine()->addPluginPath(QCoreApplication::applicationDirPath());

在“ assets:”之后加正斜杠很重要,没有它就无法工作。 在构建文件夹中,您需要具有以下目录结构:/home/my_name/project_build_directory/qml/my_plugin。您的插件目录应具有您的.so库和qmldir文件。 然后,您需要将插件打包。通过将以下内容添加到您的.pro文件中,应该可以做到这一点:

ANDROID_EXTRA_PLUGINS += /full/path/to/your/plugin/folder

但是对于Qt 5.11的Qt Creator来说,这不适用于我,所以我要做的是手动将其添加到.json部署文件中:

"android-extra-plugins":"/full/path/to/your/plugin",

build变量可能在Qt Creator的早期版本中有效。 如果您的插件具有Qt依赖项,则androiddeployqt不会选择它们,您必须手动添加它们。这可以通过向.pro文件中添加构建变量来完成:

ANDROID_EXTRA_LIBS += /your/qt/path/lib/libQt5Xml.so
... etc

或,如果这不起作用,则必须将其手动添加到JSON构建文件中:

"android-extra-libs":"/your/qt/path/lib/libQt5Xml.so,/your/qt/path/lib/libQt5Network.so",
...etc

该过程很混乱,在对构建进行某些更改时会自动重新生成JSON文件,因此必须将更改再次粘贴到该文件。如果有人知道如何改进流程,将不胜感激。您应该在Android设备上最终看到的是,您的插件.so最终将位于所有.so库所在的同一目录中,而您的qmldir最终将会在{{1}中}。