Porting to new HMI Framework 49/11949/1 4.99.3 eel/4.99.3 eel_4.99.3
authorScott Murray <scott.murray@konsulko.com>
Tue, 14 Nov 2017 13:33:09 +0000 (22:33 +0900)
committerScott Murray <scott.murray@konsulko.com>
Tue, 14 Nov 2017 13:33:09 +0000 (22:33 +0900)
Changes following guidelines given in "Kickstart: New HMI Framework"
document at:

https://wiki.automotivelinux.org/_media/kickstart_apps_migration_guide.pdf

Change-Id: Iacfb8ad1a6851d0ca9bbcc1815ef5f0822b3836c
Signed-off-by: Scott Murray <scott.murray@konsulko.com>
app/app.pro
app/main.cpp
package/config.xml

index 34c37f4..a33fc0d 100644 (file)
@@ -10,7 +10,7 @@ SOURCES = main.cpp \
     paclient.cpp
 
 CONFIG += link_pkgconfig
-PKGCONFIG += libpulse
+PKGCONFIG += libhomescreen qlibwindowmanager libpulse
 
 RESOURCES += \
     Mixer.qrc
index 1868ac6..e93cf0c 100644 (file)
 #include "pacontrolmodel.h"
 
 #include <QtCore/QDebug>
+#include <QtCore/QCommandLineParser>
+#include <QtCore/QUrlQuery>
 #include <QtCore/QDir>
 #include <QtCore/QStandardPaths>
 #include <QtCore/QThread>
 #include <QtGui/QGuiApplication>
 #include <QtQml/QQmlApplicationEngine>
 #include <QtQml/QQmlContext>
-#include <QtQml/qqml.h>
 #include <QtQuickControls2/QQuickStyle>
+#include <QtQml/qqml.h>
 #include <QtQuick/qquickitem.h>
 #include <QtQuick/qquickview.h>
-
-
-#ifdef HAVE_LIBHOMESCREEN
+#include <QQuickWindow>
 #include <libhomescreen.hpp>
-#endif
+#include <qlibwindowmanager.h>
 
 int main(int argc, char *argv[])
 {
-#ifdef HAVE_LIBHOMESCREEN
-       LibHomeScreen libHomeScreen;
-
-       if (!libHomeScreen.renderAppToAreaAllowed(0, 1)) {
-               qWarning() << "renderAppToAreaAllowed is denied";
-               return -1;
-       }
-#endif
+       QString myname = QString("Mixer");
 
        QGuiApplication app(argc, argv);
 
        QQuickStyle::setStyle("AGL");
 
+       QCommandLineParser parser;
+       parser.addPositionalArgument("port", app.translate("main", "port for binding"));
+       parser.addPositionalArgument("secret", app.translate("main", "secret for binding"));
+       parser.addHelpOption();
+       parser.addVersionOption();
+       parser.process(app);
+       QStringList positionalArguments = parser.positionalArguments();
+
        // Fire up PA client QThread
        QThread* pat = new QThread;
        PaClient* client = new PaClient();
@@ -60,20 +61,72 @@ int main(int argc, char *argv[])
        qmlRegisterType<PaControlModel>("PaControlModel", 1, 0, "PaControlModel");
 
        QQmlApplicationEngine engine;
-       engine.load(QUrl(QStringLiteral("qrc:/Mixer.qml")));
+       if (positionalArguments.length() == 2) {
+               int port = positionalArguments.takeFirst().toInt();
+               QString secret = positionalArguments.takeFirst();
+               QUrl bindingAddress;
+               bindingAddress.setScheme(QStringLiteral("ws"));
+               bindingAddress.setHost(QStringLiteral("localhost"));
+               bindingAddress.setPort(port);
+               bindingAddress.setPath(QStringLiteral("/api"));
+               QUrlQuery query;
+               query.addQueryItem(QStringLiteral("token"), secret);
+               bindingAddress.setQuery(query);
+               QQmlContext *context = engine.rootContext();
+               context->setContextProperty(QStringLiteral("bindingAddress"), bindingAddress);
 
-       // Find the instantiated model QObject and connect the signals/slots
-       QList<QObject *> mobjs = engine.rootObjects();
-       PaControlModel *pacm = mobjs.first()->findChild<PaControlModel *>("pacm");
-       QObject::connect(client, SIGNAL(controlAdded(int, QString, int, int, const char *, int)),
-                        pacm, SLOT(addOneControl(int, QString, int, int, const char *, int)));
-       QObject::connect(client, SIGNAL(volumeExternallyChanged(uint32_t, uint32_t, uint32_t, uint32_t)),
-                        pacm, SLOT(changeExternalVolume(uint32_t, uint32_t, uint32_t, uint32_t)));
-       QObject::connect(pacm, SIGNAL(volumeChanged(uint32_t, uint32_t, uint32_t, uint32_t)),
-                       client, SLOT(setVolume(uint32_t, uint32_t, uint32_t, uint32_t)));
+               std::string token = secret.toStdString();
+               LibHomeScreen* hs = new LibHomeScreen();
+               QLibWindowmanager* qwm = new QLibWindowmanager();
 
-       // Initalize PA client
-       client->init();
+               // WindowManager
+               if(qwm->init(port,secret) != 0){
+                       exit(EXIT_FAILURE);
+               }
+               // Request a surface as described in layers.json windowmanager’s file
+               if (qwm->requestSurface(myname) != 0) {
+                       exit(EXIT_FAILURE);
+               }
+               // Create an event callbnewack against an event type. Here a lambda is called when SyncDraw event occurs
+               qwm->set_event_handler(QLibWindowmanager::Event_SyncDraw, [qwm, myname](json_object *object) {
+                               fprintf(stderr, "Surface got syncDraw!\n");
+                               qwm->endDraw(myname);
+                       });
 
+               // HomeScreen
+               hs->init(port, token.c_str());
+               // Set the event handler for Event_TapShortcut which will activate the surface for windowmanager
+               hs->set_event_handler(LibHomeScreen::Event_TapShortcut, [qwm, myname](json_object *object){
+                       json_object *appnameJ = nullptr;
+                       if(json_object_object_get_ex(object, "application_name", &appnameJ))
+                       {
+                               const char *appname = json_object_get_string(appnameJ);
+                               if(myname == appname)
+                               {
+                                       qDebug("Surface %s got tapShortcut\n", appname);
+                                       qwm->activateSurface(myname);
+                               }
+                       }
+               });
+
+               engine.load(QUrl(QStringLiteral("qrc:/Mixer.qml")));
+
+               // Find the instantiated model QObject and connect the signals/slots
+               QList<QObject *> mobjs = engine.rootObjects();
+
+               QQuickWindow *window = qobject_cast<QQuickWindow *>(mobjs.first());
+               QObject::connect(window, SIGNAL(frameSwapped()), qwm, SLOT(slotActivateSurface()));
+
+               PaControlModel *pacm = mobjs.first()->findChild<PaControlModel *>("pacm");
+               QObject::connect(client, SIGNAL(controlAdded(int, QString, int, int, const char *, int)),
+                                pacm, SLOT(addOneControl(int, QString, int, int, const char *, int)));
+               QObject::connect(client, SIGNAL(volumeExternallyChanged(uint32_t, uint32_t, uint32_t, uint32_t)),
+                                pacm, SLOT(changeExternalVolume(uint32_t, uint32_t, uint32_t, uint32_t)));
+               QObject::connect(pacm, SIGNAL(volumeChanged(uint32_t, uint32_t, uint32_t, uint32_t)),
+                                client, SLOT(setVolume(uint32_t, uint32_t, uint32_t, uint32_t)));
+
+               // Initalize PA client
+               client->init();
+       }
        return app.exec();
 }
index 833cf86..bee25a0 100644 (file)
@@ -2,10 +2,17 @@
 <widget xmlns="http://www.w3.org/ns/widgets" id="mixer" version="0.1">
   <name>Mixer</name>
   <icon src="icon.svg"/>
-  <content src="bin/mixer" type="application/x-executable"/>
+  <content src="bin/mixer" type="application/vnd.agl.native"/>
   <description>Simple PulseAudio mixer app</description>
   <author>Matt Porter &lt;mporter@konsulko.com&gt;</author>
   <license>APL 2.0</license>
+  <feature name="urn:AGL:widget:required-api">
+    <param name="windowmanager" value="ws" />
+    <param name="homescreen" value="ws" />
+  </feature>
+  <feature name="urn:AGL:widget:required-permission">
+    <param name="urn:AGL:permission::public:no-htdocs" value="required" />
+  </feature>
 </widget>