我正在尝试从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
答案 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中发现了一些丑陋的东西:
Qt确实实现了某种机制来提供&#34; searchpath&#34; (不是URL)喜欢&#34; qrc:&#34;,&#34; assets:&#34;,其中内核是QAbstractFileEngineHandler。
QAbstractFileEngineHandler在Qt5中不再公开,只提供QFile,QFileInfo,QDir,QIcon和QFile提供的东西的摘要,包括QImage,以及专门用于Android资产的QAndroidAssetsFileEngineHandler。
所以SearchPath架构与QUrl不兼容,即使它们看起来如此相似,因为你知道QtQuick是setSource到某个URL,所以qml文件不会被&#34; assets:&#34;直接前缀。
您可以在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上进行了测试
此外,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}中}。