Platform specific part of QML-file?
I want a qml-object to only show up if my application is compiled for mac os. How do I do that?
10 replies
Not that I know of, but it would be quite simple I think:
- //header
- {
- Q_OBJECT
- Q_PROPERTY (bool isMac READ isMac)
- public:
- bool isMac();
- }
- //implementation:
- {
- }
- bool PlatformDetails::isMac()
- {
- #ifdef Q_OS_MAC
- return true;
- #elif
- return false;
- #endif
- }
And then you simply create an instance an insert it into your QML context. Of course, you can make the class a bit more useful by returning an enum that can represent all the platforms instead of a simple boolean for mac, but you get the idea.
Could there be some compile-time switch?
After all the QML code is wrapped into a c++ application and at the moment of compilation you do know the platform. Then a #define could include different Settings.qml for each platform. That is what I am thinking about for my own project (need some fine-tuning for Symbian, MeeGo, Android).
It would be cool to see a complete example from somebody though.
I had a similar challenge to have some variation just for Maemo (and not for Symbian). My project is pure QML, c++ part is just a thin wrapper generated by Qt Creator wizard and I wanted the platform dependence be isolated to as small piece of c++ as possible.
What I ended up with was using C macros in main.cpp to define a qml context-wide property the following way:
- #if defined(Q_OS_SYMBIAN)
- int platformId = 0;
- #elif defined(Q_WS_MAEMO_5)
- int platformId = 1;
- #elif defined(QT_SIMULATOR)
- int platformId = 2;
- #else
- // desktop probably
- int platformId = 3;
- #endif
- viewer.rootContext()->setContextProperty("platform", platformId);
- viewer.showExpanded();
- return app.exec();
Then in your qml you’ve got a platform property that can be used when doing property binding or inside a function the following way (my code returns a different set of settings for the different platform):
- function resolvePlatform() {
- switch(platform) {
- case 0:
- return setSymbian
- case 1:
- return setMaemo
- case 2:
- return setSimulator
- default:
- return setDesktop
- }
- }
Certainly this run-time platform dependency switch still means that you need to include all your files to the project regardless of whether they make sense for a platform – I didn’t find a way to use .pro syntax for replacing e.g. Settings.qml at build time.
That also means that unless you are ready to mess with Loader your qmls should compile on all the platforms you are interested in. For example, if you want to target Maemo5 as well, you need to use “import Qt 4.7”, not “import QtQuick 1.0”.
P.S.
The code above is proof of concept for now. In your real app, you will certainly use proper constants, not just numbers.
I tried playing with DEPLOYMENTFOLDERS and failed. QML files are gathered using poorly documented combo of folder_01.source, folder_01.target and DEPLOYMENTFOLDERS
Whatever I was doing I failed to put qmls from two different source folders into the same path inside a binary – they always tend to have a path specific component. Further investigation showed that this logic comes from generated qmlapplicationviewer.pri, but messing with it was too much for me.
hmm, not sure what’s not exactly working for you but platform directive + DEPLOYMENTFOLDERS seems to work for me.
something like…
- symbian{
- f3.source = ./symbian/plat
- f3.target = qml
- DEPLOYMENTFOLDERS += f3
- }
- win32{
- f3.source = ./win/plat
- f3.target = qml
- DEPLOYMENTFOLDERS += f3
- }
You’ll have qml/plat/ folder in target with files depending on the platform.
Maybe DEPLOYMENT is more flexible.
Hmm, I tried the similar code and it was resulting in two subfolders in the binary. If I remember correctly, I was getting something like: – qml/symbian/plat – qml/win/plat – qml/common (I had some common files as well)
That may be fine depending on what you want, but I wanted to put the files from both /win/plat and /symbian/plat to the same directory so that I could just replace Settings.qml and maybe some layout files.
Maybe this behavior is platform-specific. I tried on Mac OS X
You must log in to post a reply. Not a member yet? Register here!




