From 4439de240a0297c42256ac2d61ab5ae37a677032 Mon Sep 17 00:00:00 2001 From: Brad Davis Date: Thu, 22 Jun 2017 14:46:24 -0700 Subject: [PATCH] Trying to fix QtQml/QtScript crashes --- CMakeGraphvizOptions.cmake | 3 +- interface/src/Application.cpp | 18 +- interface/src/Menu.cpp | 11 +- interface/src/scripting/QmlWrapper.h | 63 --- .../scripting/ToolbarScriptingInterface.cpp | 104 ----- .../src/scripting/ToolbarScriptingInterface.h | 26 -- interface/src/ui/DialogsManager.cpp | 2 +- interface/src/ui/LoginDialog.cpp | 2 +- interface/src/ui/overlays/Web3DOverlay.cpp | 11 +- libraries/entities-renderer/CMakeLists.txt | 2 +- .../src/RenderableWebEntityItem.cpp | 25 +- .../src/RenderableWebEntityItem.h | 5 +- libraries/networking/CMakeLists.txt | 2 +- libraries/script-engine/CMakeLists.txt | 2 +- .../script-engine/src/MenuItemProperties.h | 6 +- libraries/script-engine/src/ScriptEngine.cpp | 6 +- libraries/script-engine/src/ScriptEngines.cpp | 3 - libraries/ui/CMakeLists.txt | 9 +- libraries/ui/src/OffscreenUi.cpp | 3 +- libraries/ui/src/OffscreenUi.h | 2 +- .../gl => ui/src/ui}/OffscreenQmlSurface.cpp | 44 ++- .../gl => ui/src/ui}/OffscreenQmlSurface.h | 0 .../src/ui}/OffscreenQmlSurfaceCache.cpp | 0 .../src/ui}/OffscreenQmlSurfaceCache.h | 0 libraries/ui/src/ui/QmlWrapper.cpp | 59 +++ libraries/ui/src/ui/QmlWrapper.h | 44 +++ .../src/ui}/TabletScriptingInterface.cpp | 363 +++++++++++------- .../src/ui}/TabletScriptingInterface.h | 44 ++- .../ui/src/ui/ToolbarScriptingInterface.cpp | 124 ++++++ .../ui/src/ui/ToolbarScriptingInterface.h | 56 +++ .../src/ui/types}/FileTypeProfile.cpp | 0 .../src => ui/src/ui/types}/FileTypeProfile.h | 0 .../ui/types}/FileTypeRequestInterceptor.cpp | 0 .../ui/types}/FileTypeRequestInterceptor.h | 0 .../ui/types}/HFTabletWebEngineProfile.cpp | 0 .../src/ui/types}/HFTabletWebEngineProfile.h | 0 .../HFTabletWebEngineRequestInterceptor.cpp | 0 .../HFTabletWebEngineRequestInterceptor.h | 0 .../src/ui/types}/HFWebEngineProfile.cpp | 0 .../src/ui/types}/HFWebEngineProfile.h | 0 .../types}/HFWebEngineRequestInterceptor.cpp | 0 .../ui/types}/HFWebEngineRequestInterceptor.h | 0 .../src/ui/types}/RequestFilters.cpp | 0 .../src => ui/src/ui/types}/RequestFilters.h | 0 .../src => ui/src/ui/types}/SoundEffect.cpp | 2 - .../src => ui/src/ui/types}/SoundEffect.h | 0 46 files changed, 618 insertions(+), 423 deletions(-) delete mode 100644 interface/src/scripting/QmlWrapper.h delete mode 100644 interface/src/scripting/ToolbarScriptingInterface.cpp delete mode 100644 interface/src/scripting/ToolbarScriptingInterface.h rename libraries/{gl/src/gl => ui/src/ui}/OffscreenQmlSurface.cpp (97%) rename libraries/{gl/src/gl => ui/src/ui}/OffscreenQmlSurface.h (100%) rename libraries/{gl/src/gl => ui/src/ui}/OffscreenQmlSurfaceCache.cpp (100%) rename libraries/{gl/src/gl => ui/src/ui}/OffscreenQmlSurfaceCache.h (100%) create mode 100644 libraries/ui/src/ui/QmlWrapper.cpp create mode 100644 libraries/ui/src/ui/QmlWrapper.h rename libraries/{script-engine/src => ui/src/ui}/TabletScriptingInterface.cpp (69%) rename libraries/{script-engine/src => ui/src/ui}/TabletScriptingInterface.h (90%) create mode 100644 libraries/ui/src/ui/ToolbarScriptingInterface.cpp create mode 100644 libraries/ui/src/ui/ToolbarScriptingInterface.h rename libraries/{networking/src => ui/src/ui/types}/FileTypeProfile.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/FileTypeProfile.h (100%) rename libraries/{networking/src => ui/src/ui/types}/FileTypeRequestInterceptor.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/FileTypeRequestInterceptor.h (100%) rename libraries/{networking/src => ui/src/ui/types}/HFTabletWebEngineProfile.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/HFTabletWebEngineProfile.h (100%) rename libraries/{networking/src => ui/src/ui/types}/HFTabletWebEngineRequestInterceptor.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/HFTabletWebEngineRequestInterceptor.h (100%) rename libraries/{networking/src => ui/src/ui/types}/HFWebEngineProfile.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/HFWebEngineProfile.h (100%) rename libraries/{networking/src => ui/src/ui/types}/HFWebEngineRequestInterceptor.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/HFWebEngineRequestInterceptor.h (100%) rename libraries/{networking/src => ui/src/ui/types}/RequestFilters.cpp (100%) rename libraries/{networking/src => ui/src/ui/types}/RequestFilters.h (100%) rename libraries/{script-engine/src => ui/src/ui/types}/SoundEffect.cpp (97%) rename libraries/{script-engine/src => ui/src/ui/types}/SoundEffect.h (100%) diff --git a/CMakeGraphvizOptions.cmake b/CMakeGraphvizOptions.cmake index f1b73f1cae..3b23172a79 100644 --- a/CMakeGraphvizOptions.cmake +++ b/CMakeGraphvizOptions.cmake @@ -1 +1,2 @@ -set(GRAPHVIZ_EXTERNAL_LIBS FALSE) \ No newline at end of file +set(GRAPHVIZ_EXTERNAL_LIBS FALSE) +set(GRAPHVIZ_IGNORE_TARGETS "shared;networking") \ No newline at end of file diff --git a/interface/src/Application.cpp b/interface/src/Application.cpp index e12c73d0bc..ccf81ace3f 100644 --- a/interface/src/Application.cpp +++ b/interface/src/Application.cpp @@ -99,8 +99,8 @@ #include #include #include -#include #include +#include #include #include #include @@ -125,7 +125,8 @@ #include #include #include -#include +#include +#include #include #include #include @@ -166,7 +167,6 @@ #include "scripting/SettingsScriptingInterface.h" #include "scripting/WindowScriptingInterface.h" #include "scripting/ControllerScriptingInterface.h" -#include "scripting/ToolbarScriptingInterface.h" #include "scripting/RatesScriptingInterface.h" #if defined(Q_OS_MAC) || defined(Q_OS_WIN) #include "SpeechRecognizer.h" @@ -1792,6 +1792,8 @@ void Application::cleanupBeforeQuit() { #endif // stop QML + DependencyManager::destroy(); + DependencyManager::destroy(); DependencyManager::destroy(); DependencyManager::destroy(); @@ -5535,8 +5537,16 @@ void Application::registerScriptEngineWithApplicationServices(ScriptEngine* scri scriptEngine->registerGlobalObject("OffscreenFlags", DependencyManager::get()->getFlags()); scriptEngine->registerGlobalObject("Desktop", DependencyManager::get().data()); + + qScriptRegisterMetaType(scriptEngine, wrapperToScriptValue, wrapperFromScriptValue); + qScriptRegisterMetaType(scriptEngine, wrapperToScriptValue, wrapperFromScriptValue); scriptEngine->registerGlobalObject("Toolbars", DependencyManager::get().data()); + qScriptRegisterMetaType(scriptEngine, wrapperToScriptValue, wrapperFromScriptValue); + qScriptRegisterMetaType(scriptEngine, wrapperToScriptValue, wrapperFromScriptValue); + scriptEngine->registerGlobalObject("Tablet", DependencyManager::get().data()); + + DependencyManager::get().data()->setToolbarScriptingInterface(DependencyManager::get().data()); scriptEngine->registerGlobalObject("Window", DependencyManager::get().data()); @@ -5610,7 +5620,6 @@ void Application::registerScriptEngineWithApplicationServices(ScriptEngine* scri scriptEngine->registerGlobalObject("EntityScriptServerLog", entityScriptServerLog.data()); scriptEngine->registerGlobalObject("AvatarInputs", AvatarInputs::getInstance()); - qScriptRegisterMetaType(scriptEngine, OverlayIDtoScriptValue, OverlayIDfromScriptValue); // connect this script engines printedMessage signal to the global ScriptEngines these various messages @@ -5619,6 +5628,7 @@ void Application::registerScriptEngineWithApplicationServices(ScriptEngine* scri connect(scriptEngine, &ScriptEngine::warningMessage, DependencyManager::get().data(), &ScriptEngines::onWarningMessage); connect(scriptEngine, &ScriptEngine::infoMessage, DependencyManager::get().data(), &ScriptEngines::onInfoMessage); connect(scriptEngine, &ScriptEngine::clearDebugWindow, DependencyManager::get().data(), &ScriptEngines::onClearDebugWindow); + } bool Application::canAcceptURL(const QString& urlString) const { diff --git a/interface/src/Menu.cpp b/interface/src/Menu.cpp index 8c9baa7c43..5fbae75a57 100644 --- a/interface/src/Menu.cpp +++ b/interface/src/Menu.cpp @@ -19,7 +19,7 @@ #include #include #include -#include +#include #include #include #include @@ -121,9 +121,14 @@ Menu::Menu() { QAction::NoRole, UNSPECIFIED_POSITION, "Advanced"); // Edit > Reload All Scripts... [advanced] - addActionToQMenuAndActionHash(editMenu, MenuOption::ReloadAllScripts, Qt::CTRL | Qt::Key_R, - scriptEngines.data(), SLOT(reloadAllScripts()), + action = addActionToQMenuAndActionHash(editMenu, MenuOption::ReloadAllScripts, Qt::CTRL | Qt::Key_R, + nullptr, nullptr, QAction::NoRole, UNSPECIFIED_POSITION, "Advanced"); + connect(action, &QAction::triggered, [] { + DependencyManager::get()->reloadAllScripts(); + DependencyManager::get()->clearCache(); + }); + // Edit > Console... [advanced] addActionToQMenuAndActionHash(editMenu, MenuOption::Console, Qt::CTRL | Qt::ALT | Qt::Key_J, diff --git a/interface/src/scripting/QmlWrapper.h b/interface/src/scripting/QmlWrapper.h deleted file mode 100644 index 7dd319e445..0000000000 --- a/interface/src/scripting/QmlWrapper.h +++ /dev/null @@ -1,63 +0,0 @@ -// -// Created by Anthony J. Thibault on 2016-12-12 -// Copyright 2013-2016 High Fidelity, Inc. -// -// Distributed under the Apache License, Version 2.0. -// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html -// - -#ifndef hifi_QmlWrapper_h -#define hifi_QmlWrapper_h - -#include -#include -#include - -class QmlWrapper : public QObject { - Q_OBJECT -public: - QmlWrapper(QObject* qmlObject, QObject* parent = nullptr) - : QObject(parent), _qmlObject(qmlObject) { - } - - Q_INVOKABLE void writeProperty(QString propertyName, QVariant propertyValue) { - auto offscreenUi = DependencyManager::get(); - offscreenUi->executeOnUiThread([=] { - _qmlObject->setProperty(propertyName.toStdString().c_str(), propertyValue); - }); - } - - Q_INVOKABLE void writeProperties(QVariant propertyMap) { - auto offscreenUi = DependencyManager::get(); - offscreenUi->executeOnUiThread([=] { - QVariantMap map = propertyMap.toMap(); - for (const QString& key : map.keys()) { - _qmlObject->setProperty(key.toStdString().c_str(), map[key]); - } - }); - } - - Q_INVOKABLE QVariant readProperty(const QString& propertyName) { - auto offscreenUi = DependencyManager::get(); - return offscreenUi->returnFromUiThread([&]()->QVariant { - return _qmlObject->property(propertyName.toStdString().c_str()); - }); - } - - Q_INVOKABLE QVariant readProperties(const QVariant& propertyList) { - auto offscreenUi = DependencyManager::get(); - return offscreenUi->returnFromUiThread([&]()->QVariant { - QVariantMap result; - for (const QVariant& property : propertyList.toList()) { - QString propertyString = property.toString(); - result.insert(propertyString, _qmlObject->property(propertyString.toStdString().c_str())); - } - return result; - }); - } - -protected: - QObject* _qmlObject{ nullptr }; -}; - -#endif \ No newline at end of file diff --git a/interface/src/scripting/ToolbarScriptingInterface.cpp b/interface/src/scripting/ToolbarScriptingInterface.cpp deleted file mode 100644 index 2b4f64f35e..0000000000 --- a/interface/src/scripting/ToolbarScriptingInterface.cpp +++ /dev/null @@ -1,104 +0,0 @@ -// -// Created by Bradley Austin Davis on 2016-06-16 -// Copyright 2013-2016 High Fidelity, Inc. -// -// Distributed under the Apache License, Version 2.0. -// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html -// - -#include "ToolbarScriptingInterface.h" - - -#include - -#include -#include "QmlWrapper.h" - -class ToolbarButtonProxy : public QmlWrapper { - Q_OBJECT - -public: - ToolbarButtonProxy(QObject* qmlObject, QObject* parent = nullptr) : QmlWrapper(qmlObject, parent) { - std::lock_guard guard(_mutex); - _qmlButton = qobject_cast(qmlObject); - connect(qmlObject, SIGNAL(clicked()), this, SIGNAL(clicked())); - } - - Q_INVOKABLE void editProperties(QVariantMap properties) { - std::lock_guard guard(_mutex); - QVariantMap::const_iterator iter = properties.constBegin(); - while (iter != properties.constEnd()) { - _properties[iter.key()] = iter.value(); - if (_qmlButton) { - // [01/25 14:26:20] [WARNING] [default] QMetaObject::invokeMethod: No such method ToolbarButton_QMLTYPE_195::changeProperty(QVariant,QVariant) - - QMetaObject::invokeMethod(_qmlButton, "changeProperty", Qt::AutoConnection, - Q_ARG(QVariant, QVariant(iter.key())), Q_ARG(QVariant, iter.value())); - } - ++iter; - } - } - -signals: - void clicked(); - -protected: - mutable std::mutex _mutex; - QQuickItem* _qmlButton { nullptr }; - QVariantMap _properties; -}; - -class ToolbarProxy : public QmlWrapper { - Q_OBJECT - -public: - ToolbarProxy(QObject* qmlObject, QObject* parent = nullptr) : QmlWrapper(qmlObject, parent) { } - - Q_INVOKABLE QObject* addButton(const QVariant& properties) { - QVariant resultVar; - Qt::ConnectionType connectionType = Qt::AutoConnection; - if (QThread::currentThread() != _qmlObject->thread()) { - connectionType = Qt::BlockingQueuedConnection; - } - bool invokeResult = QMetaObject::invokeMethod(_qmlObject, "addButton", connectionType, Q_RETURN_ARG(QVariant, resultVar), Q_ARG(QVariant, properties)); - if (!invokeResult) { - return nullptr; - } - - QObject* rawButton = qvariant_cast(resultVar); - if (!rawButton) { - return nullptr; - } - - return new ToolbarButtonProxy(rawButton, this); - } - - Q_INVOKABLE void removeButton(const QVariant& name) { - QMetaObject::invokeMethod(_qmlObject, "removeButton", Qt::AutoConnection, Q_ARG(QVariant, name)); - } -}; - - -QObject* ToolbarScriptingInterface::getToolbar(const QString& toolbarId) { - auto offscreenUi = DependencyManager::get(); - auto desktop = offscreenUi->getDesktop(); - Qt::ConnectionType connectionType = Qt::AutoConnection; - if (QThread::currentThread() != desktop->thread()) { - connectionType = Qt::BlockingQueuedConnection; - } - QVariant resultVar; - bool invokeResult = QMetaObject::invokeMethod(desktop, "getToolbar", connectionType, Q_RETURN_ARG(QVariant, resultVar), Q_ARG(QVariant, toolbarId)); - if (!invokeResult) { - return nullptr; - } - - QObject* rawToolbar = qvariant_cast(resultVar); - if (!rawToolbar) { - return nullptr; - } - - return new ToolbarProxy(rawToolbar); -} - - -#include "ToolbarScriptingInterface.moc" diff --git a/interface/src/scripting/ToolbarScriptingInterface.h b/interface/src/scripting/ToolbarScriptingInterface.h deleted file mode 100644 index 9379284e55..0000000000 --- a/interface/src/scripting/ToolbarScriptingInterface.h +++ /dev/null @@ -1,26 +0,0 @@ -// -// Created by Bradley Austin Davis on 2016-06-16 -// Copyright 2013-2016 High Fidelity, Inc. -// -// Distributed under the Apache License, Version 2.0. -// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html -// - -#ifndef hifi_ToolbarScriptingInterface_h -#define hifi_ToolbarScriptingInterface_h - -#include - -#include - -#include - -class ToolbarProxy; - -class ToolbarScriptingInterface : public QObject, public Dependency { - Q_OBJECT -public: - Q_INVOKABLE QObject* getToolbar(const QString& toolbarId); -}; - -#endif // hifi_ToolbarScriptingInterface_h diff --git a/interface/src/ui/DialogsManager.cpp b/interface/src/ui/DialogsManager.cpp index 96b5da3a55..f216bb4edc 100644 --- a/interface/src/ui/DialogsManager.cpp +++ b/interface/src/ui/DialogsManager.cpp @@ -17,6 +17,7 @@ #include #include #include +#include #include "AddressBarDialog.h" #include "ConnectionFailureDialog.h" @@ -28,7 +29,6 @@ #include "PreferencesDialog.h" #include "UpdateDialog.h" -#include "TabletScriptingInterface.h" #include "scripting/HMDScriptingInterface.h" static const QVariant TABLET_ADDRESS_DIALOG = "TabletAddressDialog.qml"; diff --git a/interface/src/ui/LoginDialog.cpp b/interface/src/ui/LoginDialog.cpp index 10783afd23..c9341fd3d9 100644 --- a/interface/src/ui/LoginDialog.cpp +++ b/interface/src/ui/LoginDialog.cpp @@ -19,13 +19,13 @@ #include #include #include +#include #include "AccountManager.h" #include "DependencyManager.h" #include "Menu.h" #include "Application.h" -#include "TabletScriptingInterface.h" #include "scripting/HMDScriptingInterface.h" HIFI_QML_DEF(LoginDialog) diff --git a/interface/src/ui/overlays/Web3DOverlay.cpp b/interface/src/ui/overlays/Web3DOverlay.cpp index b67ca06aa3..80f11fa552 100644 --- a/interface/src/ui/overlays/Web3DOverlay.cpp +++ b/interface/src/ui/overlays/Web3DOverlay.cpp @@ -25,16 +25,15 @@ #include #include #include -#include +#include +#include +#include #include #include -#include #include #include #include #include -#include -#include #include #include "scripting/AccountScriptingInterface.h" #include "scripting/HMDScriptingInterface.h" @@ -87,7 +86,7 @@ Web3DOverlay::~Web3DOverlay() { if (rootItem && rootItem->objectName() == "tabletRoot") { auto tabletScriptingInterface = DependencyManager::get(); - tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", nullptr, nullptr); + tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", nullptr); } // Fix for crash in QtWebEngineCore when rapidly switching domains @@ -206,7 +205,7 @@ void Web3DOverlay::loadSourceURL() { _webSurface->getSurfaceContext()->setContextProperty("SoundCache", DependencyManager::get().data()); _webSurface->getSurfaceContext()->setContextProperty("pathToFonts", "../../"); - tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", _webSurface->getRootItem(), _webSurface.data()); + tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", _webSurface.data()); // mark the TabletProxy object as cpp ownership. QObject* tablet = tabletScriptingInterface->getTablet("com.highfidelity.interface.tablet.system"); diff --git a/libraries/entities-renderer/CMakeLists.txt b/libraries/entities-renderer/CMakeLists.txt index 3df33b6a5d..40111e257b 100644 --- a/libraries/entities-renderer/CMakeLists.txt +++ b/libraries/entities-renderer/CMakeLists.txt @@ -1,7 +1,7 @@ set(TARGET_NAME entities-renderer) AUTOSCRIBE_SHADER_LIB(gpu model procedural render render-utils) setup_hifi_library(Widgets Network Script) -link_hifi_libraries(shared gpu procedural model model-networking script-engine render render-utils image) +link_hifi_libraries(shared gpu procedural model model-networking script-engine render render-utils image ui) include_hifi_library_headers(networking) include_hifi_library_headers(gl) include_hifi_library_headers(ktx) diff --git a/libraries/entities-renderer/src/RenderableWebEntityItem.cpp b/libraries/entities-renderer/src/RenderableWebEntityItem.cpp index 948a219831..9a898b4071 100644 --- a/libraries/entities-renderer/src/RenderableWebEntityItem.cpp +++ b/libraries/entities-renderer/src/RenderableWebEntityItem.cpp @@ -23,7 +23,7 @@ #include #include #include -#include +#include #include "EntityTreeRenderer.h" #include "EntitiesRendererLogging.h" @@ -74,19 +74,6 @@ bool RenderableWebEntityItem::buildWebSurface(QSharedPointer qWarning() << "Too many concurrent web views to create new view"; return false; } - QString javaScriptToInject; - QFile webChannelFile(":qtwebchannel/qwebchannel.js"); - QFile createGlobalEventBridgeFile(PathUtils::resourcesPath() + "/html/createGlobalEventBridge.js"); - if (webChannelFile.open(QFile::ReadOnly | QFile::Text) && - createGlobalEventBridgeFile.open(QFile::ReadOnly | QFile::Text)) { - QString webChannelStr = QTextStream(&webChannelFile).readAll(); - QString createGlobalEventBridgeStr = QTextStream(&createGlobalEventBridgeFile).readAll(); - - // concatenate these js files - _javaScriptToInject = webChannelStr + createGlobalEventBridgeStr; - } else { - qCWarning(entitiesrenderer) << "unable to find qwebchannel.js or createGlobalEventBridge.js"; - } // Save the original GL context, because creating a QML surface will create a new context QOpenGLContext* currentContext = QOpenGLContext::currentContext(); @@ -266,10 +253,7 @@ void RenderableWebEntityItem::loadSourceURL() { _webSurface->setMaxFps(DEFAULT_MAX_FPS); } - _webSurface->load("WebEntityView.qml", [&](QQmlContext* context, QObject* obj) { - context->setContextProperty("eventBridgeJavaScriptToInject", QVariant(_javaScriptToInject)); - }); - + _webSurface->load("WebEntityView.qml"); _webSurface->getRootItem()->setProperty("url", _sourceUrl); _webSurface->getSurfaceContext()->setContextProperty("desktop", QVariant()); @@ -280,8 +264,7 @@ void RenderableWebEntityItem::loadSourceURL() { if (_webSurface->getRootItem() && _webSurface->getRootItem()->objectName() == "tabletRoot") { auto tabletScriptingInterface = DependencyManager::get(); - tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", - _webSurface->getRootItem(), _webSurface.data()); + tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", _webSurface.data()); } } _webSurface->getSurfaceContext()->setContextProperty("globalPosition", vec3toVariant(getPosition())); @@ -386,7 +369,7 @@ void RenderableWebEntityItem::destroyWebSurface() { if (rootItem && rootItem->objectName() == "tabletRoot") { auto tabletScriptingInterface = DependencyManager::get(); - tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", nullptr, nullptr); + tabletScriptingInterface->setQmlTabletRoot("com.highfidelity.interface.tablet.system", nullptr); } // Fix for crash in QtWebEngineCore when rapidly switching domains diff --git a/libraries/entities-renderer/src/RenderableWebEntityItem.h b/libraries/entities-renderer/src/RenderableWebEntityItem.h index bbd59a03d6..0f5d307766 100644 --- a/libraries/entities-renderer/src/RenderableWebEntityItem.h +++ b/libraries/entities-renderer/src/RenderableWebEntityItem.h @@ -13,10 +13,9 @@ #include #include #include -#include +#include #include -#include #include "RenderableEntityItem.h" @@ -74,8 +73,6 @@ private: QMetaObject::Connection _mouseMoveConnection; QMetaObject::Connection _hoverLeaveConnection; - QString _javaScriptToInject; - enum contentType { htmlContent, qmlContent diff --git a/libraries/networking/CMakeLists.txt b/libraries/networking/CMakeLists.txt index ff9a05b959..288e98d5a5 100644 --- a/libraries/networking/CMakeLists.txt +++ b/libraries/networking/CMakeLists.txt @@ -1,5 +1,5 @@ set(TARGET_NAME networking) -setup_hifi_library(Network WebEngine) +setup_hifi_library(Network) link_hifi_libraries(shared) target_include_directories(${TARGET_NAME} PRIVATE "${CMAKE_BINARY_DIR}/includes") diff --git a/libraries/script-engine/CMakeLists.txt b/libraries/script-engine/CMakeLists.txt index 39338fd767..0c0004d132 100644 --- a/libraries/script-engine/CMakeLists.txt +++ b/libraries/script-engine/CMakeLists.txt @@ -16,6 +16,6 @@ if (NOT ANDROID) endif () -link_hifi_libraries(shared networking octree gpu ui procedural model model-networking ktx recording avatars fbx entities controllers animation audio physics image) +link_hifi_libraries(shared networking octree gpu procedural model model-networking ktx recording avatars fbx entities controllers animation audio physics image) # ui includes gl, but link_hifi_libraries does not use transitive includes, so gl must be explicit include_hifi_library_headers(gl) diff --git a/libraries/script-engine/src/MenuItemProperties.h b/libraries/script-engine/src/MenuItemProperties.h index 9b95fc66e1..9de4337838 100644 --- a/libraries/script-engine/src/MenuItemProperties.h +++ b/libraries/script-engine/src/MenuItemProperties.h @@ -14,7 +14,6 @@ #include -#include #include "KeyEvent.h" @@ -35,7 +34,7 @@ public: QKeySequence shortcutKeySequence; // this is what we actually use, it's set from one of the above // location related items: in order of priority - int position { ui::Menu::UNSPECIFIED_POSITION }; + int position { UNSPECIFIED_POSITION }; QString beforeItem; QString afterItem; @@ -45,6 +44,9 @@ public: bool isSeparator { false }; QString grouping; /// Either: "", "Advanced", or "Developer" + +private: + static const int UNSPECIFIED_POSITION = -1; }; Q_DECLARE_METATYPE(MenuItemProperties) QScriptValue menuItemPropertiesToScriptValue(QScriptEngine* engine, const MenuItemProperties& props); diff --git a/libraries/script-engine/src/ScriptEngine.cpp b/libraries/script-engine/src/ScriptEngine.cpp index 67b16df1ce..e41d41ee09 100644 --- a/libraries/script-engine/src/ScriptEngine.cpp +++ b/libraries/script-engine/src/ScriptEngine.cpp @@ -24,6 +24,8 @@ #include #include +#include +#include #include #include @@ -48,7 +50,6 @@ #include #include #include -#include #include #include @@ -70,7 +71,6 @@ #include "WebSocketClass.h" #include "RecordingScriptingInterface.h" #include "ScriptEngines.h" -#include "TabletScriptingInterface.h" #include "ModelScriptingInterface.h" @@ -697,8 +697,6 @@ void ScriptEngine::init() { // constants globalObject().setProperty("TREE_SCALE", newVariant(QVariant(TREE_SCALE))); - registerGlobalObject("Tablet", DependencyManager::get().data()); - qScriptRegisterMetaType(this, tabletToScriptValue, tabletFromScriptValue); registerGlobalObject("Assets", &_assetScriptingInterface); registerGlobalObject("Resources", DependencyManager::get().data()); diff --git a/libraries/script-engine/src/ScriptEngines.cpp b/libraries/script-engine/src/ScriptEngines.cpp index 69de067d10..4f134d8a22 100644 --- a/libraries/script-engine/src/ScriptEngines.cpp +++ b/libraries/script-engine/src/ScriptEngines.cpp @@ -16,8 +16,6 @@ #include #include -#include - #include "ScriptEngine.h" #include "ScriptEngineLogging.h" @@ -431,7 +429,6 @@ void ScriptEngines::setScriptsLocation(const QString& scriptsLocation) { void ScriptEngines::reloadAllScripts() { qCDebug(scriptengine) << "reloadAllScripts -- clearing caches"; DependencyManager::get()->clearCache(); - DependencyManager::get()->clearCache(); qCDebug(scriptengine) << "reloadAllScripts -- stopping all scripts"; stopAllScripts(true); } diff --git a/libraries/ui/CMakeLists.txt b/libraries/ui/CMakeLists.txt index f2b48446fe..68a6fd25b9 100644 --- a/libraries/ui/CMakeLists.txt +++ b/libraries/ui/CMakeLists.txt @@ -1,3 +1,8 @@ set(TARGET_NAME ui) -setup_hifi_library(OpenGL Network Qml Quick Script WebChannel WebSockets XmlPatterns) -link_hifi_libraries(shared networking gl script-engine) +setup_hifi_library(OpenGL Network Qml Quick Script WebChannel WebEngine WebSockets XmlPatterns) +link_hifi_libraries(shared networking gl audio) + +if (NOT ANDROID) + # Required for some low level GL interaction in the OffscreenQMLSurface + target_glew() +endif () diff --git a/libraries/ui/src/OffscreenUi.cpp b/libraries/ui/src/OffscreenUi.cpp index a80105293b..76a8a780b9 100644 --- a/libraries/ui/src/OffscreenUi.cpp +++ b/libraries/ui/src/OffscreenUi.cpp @@ -20,7 +20,8 @@ #include #include #include -#include + +#include "ui/TabletScriptingInterface.h" #include "FileDialogHelper.h" #include "VrMenu.h" diff --git a/libraries/ui/src/OffscreenUi.h b/libraries/ui/src/OffscreenUi.h index 55fb8b2c3d..f228f28d5e 100644 --- a/libraries/ui/src/OffscreenUi.h +++ b/libraries/ui/src/OffscreenUi.h @@ -21,9 +21,9 @@ #include #include -#include #include +#include "ui/OffscreenQmlSurface.h" #include "OffscreenQmlElement.h" class VrMenu; diff --git a/libraries/gl/src/gl/OffscreenQmlSurface.cpp b/libraries/ui/src/ui/OffscreenQmlSurface.cpp similarity index 97% rename from libraries/gl/src/gl/OffscreenQmlSurface.cpp rename to libraries/ui/src/ui/OffscreenQmlSurface.cpp index 9dcc1d7991..f0006cb399 100644 --- a/libraries/gl/src/gl/OffscreenQmlSurface.cpp +++ b/libraries/ui/src/ui/OffscreenQmlSurface.cpp @@ -6,7 +6,9 @@ // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html // #include "OffscreenQmlSurface.h" -#include "Config.h" + +// Has to come before Qt GL includes +#include #include #include @@ -33,14 +35,17 @@ #include #include #include -#include -#include -#include -#include "OffscreenGLCanvas.h" -#include "GLHelpers.h" -#include "GLLogging.h" -#include "Context.h" +#include +#include +#include + +#include "types/FileTypeProfile.h" +#include "types/HFWebEngineProfile.h" +#include "types/HFTabletWebEngineProfile.h" +#include "types/SoundEffect.h" + +#include "Logging.h" Q_LOGGING_CATEGORY(trace_render_qml, "trace.render.qml") Q_LOGGING_CATEGORY(trace_render_qml_gl, "trace.render.qml.gl") @@ -272,7 +277,7 @@ QString getEventBridgeJavascript() { QString createGlobalEventBridgeStr = QTextStream(&createGlobalEventBridgeFile).readAll(); javaScriptToInject = webChannelStr + createGlobalEventBridgeStr; } else { - qCWarning(glLogging) << "Unable to find qwebchannel.js or createGlobalEventBridge.js"; + qCWarning(uiLogging) << "Unable to find qwebchannel.js or createGlobalEventBridge.js"; } return javaScriptToInject; } @@ -297,6 +302,14 @@ private: QQmlEngine* acquireEngine(QQuickWindow* window) { Q_ASSERT(QThread::currentThread() == qApp->thread()); + if (QThread::currentThread() != qApp->thread()) { + qCWarning(uiLogging) << "Cannot acquire QML engine on any thread but the main thread"; + } + static std::once_flag once; + std::call_once(once, [] { + qmlRegisterType("Hifi", 1, 0, "SoundEffect"); + }); + if (!globalEngine) { Q_ASSERT(0 == globalEngineRefCount); globalEngine = new QQmlEngine(); @@ -325,8 +338,6 @@ QQmlEngine* acquireEngine(QQuickWindow* window) { rootContext->setContextProperty("FileTypeProfile", new FileTypeProfile(rootContext)); rootContext->setContextProperty("HFWebEngineProfile", new HFWebEngineProfile(rootContext)); rootContext->setContextProperty("HFTabletWebEngineProfile", new HFTabletWebEngineProfile(rootContext)); - - } ++globalEngineRefCount; @@ -457,7 +468,7 @@ void OffscreenQmlSurface::onAboutToQuit() { } void OffscreenQmlSurface::create(QOpenGLContext* shareContext) { - qCDebug(glLogging) << "Building QML surface"; + qCDebug(uiLogging) << "Building QML surface"; _renderControl = new QMyQuickRenderControl(); connect(_renderControl, &QQuickRenderControl::renderRequested, this, [this] { _render = true; }); @@ -548,7 +559,7 @@ void OffscreenQmlSurface::resize(const QSize& newSize_, bool forceResize) { return; } - qCDebug(glLogging) << "Offscreen UI resizing to " << newSize.width() << "x" << newSize.height(); + qCDebug(uiLogging) << "Offscreen UI resizing to " << newSize.width() << "x" << newSize.height(); gl::withSavedContext([&] { _canvas->makeCurrent(); @@ -595,6 +606,9 @@ void OffscreenQmlSurface::setBaseUrl(const QUrl& baseUrl) { } QObject* OffscreenQmlSurface::load(const QUrl& qmlSource, bool createNewContext, std::function f) { + if (QThread::currentThread() != thread()) { + qCWarning(uiLogging) << "Called load on a non-surface thread"; + } // Synchronous loading may take a while; restart the deadlock timer QMetaObject::invokeMethod(qApp, "updateHeartbeat", Qt::DirectConnection); @@ -636,7 +650,7 @@ QObject* OffscreenQmlSurface::finishQmlLoad(QQmlComponent* qmlComponent, QQmlCon disconnect(qmlComponent, &QQmlComponent::statusChanged, this, 0); if (qmlComponent->isError()) { for (const auto& error : qmlComponent->errors()) { - qCWarning(glLogging) << error.url() << error.line() << error; + qCWarning(uiLogging) << error.url() << error.line() << error; } qmlComponent->deleteLater(); return nullptr; @@ -646,7 +660,7 @@ QObject* OffscreenQmlSurface::finishQmlLoad(QQmlComponent* qmlComponent, QQmlCon QObject* newObject = qmlComponent->beginCreate(qmlContext); if (qmlComponent->isError()) { for (const auto& error : qmlComponent->errors()) { - qCWarning(glLogging) << error.url() << error.line() << error; + qCWarning(uiLogging) << error.url() << error.line() << error; } if (!_rootItem) { qFatal("Unable to finish loading QML root"); diff --git a/libraries/gl/src/gl/OffscreenQmlSurface.h b/libraries/ui/src/ui/OffscreenQmlSurface.h similarity index 100% rename from libraries/gl/src/gl/OffscreenQmlSurface.h rename to libraries/ui/src/ui/OffscreenQmlSurface.h diff --git a/libraries/gl/src/gl/OffscreenQmlSurfaceCache.cpp b/libraries/ui/src/ui/OffscreenQmlSurfaceCache.cpp similarity index 100% rename from libraries/gl/src/gl/OffscreenQmlSurfaceCache.cpp rename to libraries/ui/src/ui/OffscreenQmlSurfaceCache.cpp diff --git a/libraries/gl/src/gl/OffscreenQmlSurfaceCache.h b/libraries/ui/src/ui/OffscreenQmlSurfaceCache.h similarity index 100% rename from libraries/gl/src/gl/OffscreenQmlSurfaceCache.h rename to libraries/ui/src/ui/OffscreenQmlSurfaceCache.h diff --git a/libraries/ui/src/ui/QmlWrapper.cpp b/libraries/ui/src/ui/QmlWrapper.cpp new file mode 100644 index 0000000000..518a05613a --- /dev/null +++ b/libraries/ui/src/ui/QmlWrapper.cpp @@ -0,0 +1,59 @@ +// +// Created by Bradley Austin Davis on 2017/06/22 +// Copyright 2013-2017 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#include "QmlWrapper.h" + +#include +#include + +QmlWrapper::QmlWrapper(QObject* qmlObject, QObject* parent) + : QObject(parent), _qmlObject(qmlObject) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); +} + +void QmlWrapper::writeProperty(QString propertyName, QVariant propertyValue) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "writeProperty", Q_ARG(QString, propertyName), Q_ARG(QVariant, propertyValue)); + } + _qmlObject->setProperty(propertyName.toStdString().c_str(), propertyValue); +} + +void QmlWrapper::writeProperties(QVariant propertyMap) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "writeProperties", Q_ARG(QVariant, propertyMap)); + } + QVariantMap map = propertyMap.toMap(); + for (const QString& key : map.keys()) { + _qmlObject->setProperty(key.toStdString().c_str(), map[key]); + } +} + +QVariant QmlWrapper::readProperty(const QString& propertyName) { + if (QThread::currentThread() != thread()) { + QVariant result; + QMetaObject::invokeMethod(this, "readProperty", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QVariant, result), Q_ARG(QString, propertyName)); + return result; + } + + return _qmlObject->property(propertyName.toStdString().c_str()); +} + +QVariant QmlWrapper::readProperties(const QVariant& propertyList) { + if (QThread::currentThread() != thread()) { + QVariant result; + QMetaObject::invokeMethod(this, "readProperties", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QVariant, result), Q_ARG(QVariant, propertyList)); + return result; + } + + QVariantMap result; + for (const QVariant& property : propertyList.toList()) { + QString propertyString = property.toString(); + result.insert(propertyString, _qmlObject->property(propertyString.toStdString().c_str())); + } + return result; +} diff --git a/libraries/ui/src/ui/QmlWrapper.h b/libraries/ui/src/ui/QmlWrapper.h new file mode 100644 index 0000000000..d77e45c9dc --- /dev/null +++ b/libraries/ui/src/ui/QmlWrapper.h @@ -0,0 +1,44 @@ +// +// Created by Anthony J. Thibault on 2016-12-12 +// Copyright 2013-2016 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef hifi_QmlWrapper_h +#define hifi_QmlWrapper_h + +#include +#include +#include +#include + +class QmlWrapper : public QObject { + Q_OBJECT +public: + QmlWrapper(QObject* qmlObject, QObject* parent = nullptr); + + Q_INVOKABLE void writeProperty(QString propertyName, QVariant propertyValue); + Q_INVOKABLE void writeProperties(QVariant propertyMap); + Q_INVOKABLE QVariant readProperty(const QString& propertyName); + Q_INVOKABLE QVariant readProperties(const QVariant& propertyList); + +protected: + QObject* _qmlObject{ nullptr }; +}; + +template +QScriptValue wrapperToScriptValue(QScriptEngine* engine, T* const &in) { + if (!in) { + return engine->undefinedValue(); + } + return engine->newQObject(in, QScriptEngine::QtOwnership, QScriptEngine::ExcludeDeleteLater | QScriptEngine::ExcludeChildObjects); +} + +template +void wrapperFromScriptValue(const QScriptValue& value, T* &out) { + out = qobject_cast(value.toQObject()); +} + +#endif \ No newline at end of file diff --git a/libraries/script-engine/src/TabletScriptingInterface.cpp b/libraries/ui/src/ui/TabletScriptingInterface.cpp similarity index 69% rename from libraries/script-engine/src/TabletScriptingInterface.cpp rename to libraries/ui/src/ui/TabletScriptingInterface.cpp index ef2887fdfb..f85106918e 100644 --- a/libraries/script-engine/src/TabletScriptingInterface.cpp +++ b/libraries/ui/src/ui/TabletScriptingInterface.cpp @@ -9,66 +9,51 @@ #include "TabletScriptingInterface.h" #include +#include -#include -#include "DependencyManager.h" #include -#include -#include +#include +#include #include -#include "ScriptEngineLogging.h" -#include -#include -#include "SoundEffect.h" +#include "../QmlWindowClass.h" +#include "../OffscreenUi.h" +#include "../InfoView.h" +#include "ToolbarScriptingInterface.h" +#include "Logging.h" + +// FIXME move to global app properties const QString SYSTEM_TOOLBAR = "com.highfidelity.interface.toolbar.system"; const QString SYSTEM_TABLET = "com.highfidelity.interface.tablet.system"; -QScriptValue tabletToScriptValue(QScriptEngine* engine, TabletProxy* const &in) { - return engine->newQObject(in, QScriptEngine::QtOwnership, QScriptEngine::ExcludeDeleteLater | QScriptEngine::ExcludeChildObjects); -} - -void tabletFromScriptValue(const QScriptValue& value, TabletProxy* &out) { - out = qobject_cast(value.toQObject()); -} - TabletScriptingInterface::TabletScriptingInterface() { - qmlRegisterType("Hifi", 1, 0, "SoundEffect"); + qCDebug(uiLogging) << "Building tablet scripting interface"; } -QObject* TabletScriptingInterface::getSystemToolbarProxy() { - Qt::ConnectionType connectionType = Qt::AutoConnection; - if (QThread::currentThread() != _toolbarScriptingInterface->thread()) { - connectionType = Qt::BlockingQueuedConnection; - } +TabletScriptingInterface::~TabletScriptingInterface() { + qCDebug(uiLogging) << "Destroying tablet scripting interface"; +} - QObject* toolbarProxy = nullptr; - bool hasResult = QMetaObject::invokeMethod(_toolbarScriptingInterface, "getToolbar", connectionType, Q_RETURN_ARG(QObject*, toolbarProxy), Q_ARG(QString, SYSTEM_TOOLBAR)); - if (hasResult) { - return toolbarProxy; - } else { - qCWarning(scriptengine) << "ToolbarScriptingInterface getToolbar has no result"; - return nullptr; - } +ToolbarProxy* TabletScriptingInterface::getSystemToolbarProxy() { + Q_ASSERT(QThread::currentThread() == qApp->thread()); + return _toolbarScriptingInterface->getToolbar(SYSTEM_TOOLBAR); } TabletProxy* TabletScriptingInterface::getTablet(const QString& tabletId) { TabletProxy* tabletProxy = nullptr; - { - // the only thing guarded should be map mutation - // this avoids a deadlock with the Main thread - // from Qt::BlockingQueuedEvent invocations later in the call-tree - std::lock_guard guard(_mapMutex); + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "getTablet", Qt::BlockingQueuedConnection, Q_RETURN_ARG(TabletProxy*, tabletProxy), Q_ARG(QString, tabletId)); + return tabletProxy; + } - auto iter = _tabletProxies.find(tabletId); - if (iter != _tabletProxies.end()) { - // tablet already exists - return iter->second; - } else { - // tablet must be created - tabletProxy = new TabletProxy(this, tabletId); - _tabletProxies[tabletId] = tabletProxy; - } + auto iter = _tabletProxies.find(tabletId); + if (iter != _tabletProxies.end()) { + // tablet already exists + return iter->second; + } else { + // tablet must be created + tabletProxy = new TabletProxy(this, tabletId); + _tabletProxies[tabletId] = tabletProxy; } assert(tabletProxy); @@ -78,42 +63,40 @@ TabletProxy* TabletScriptingInterface::getTablet(const QString& tabletId) { } void TabletScriptingInterface::setToolbarMode(bool toolbarMode) { - { - // the only thing guarded should be _toolbarMode - // this avoids a deadlock with the Main thread - // from Qt::BlockingQueuedEvent invocations later in the call-tree - std::lock_guard guard(_mapMutex); - _toolbarMode = toolbarMode; - } - + Q_ASSERT(QThread::currentThread() == qApp->thread()); + _toolbarMode = toolbarMode; for (auto& iter : _tabletProxies) { iter.second->setToolbarMode(toolbarMode); } } -void TabletScriptingInterface::setQmlTabletRoot(QString tabletId, QQuickItem* qmlTabletRoot, QObject* qmlOffscreenSurface) { +void TabletScriptingInterface::setQmlTabletRoot(QString tabletId, OffscreenQmlSurface* qmlOffscreenSurface) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); TabletProxy* tablet = qobject_cast(getTablet(tabletId)); if (tablet) { - tablet->setQmlTabletRoot(qmlTabletRoot, qmlOffscreenSurface); + tablet->setQmlTabletRoot(qmlOffscreenSurface); } else { - qCWarning(scriptengine) << "TabletScriptingInterface::setupTablet() bad tablet object"; + qCWarning(uiLogging) << "TabletScriptingInterface::setupTablet() bad tablet object"; } } QQuickWindow* TabletScriptingInterface::getTabletWindow() { + Q_ASSERT(QThread::currentThread() == qApp->thread()); TabletProxy* tablet = qobject_cast(getTablet(SYSTEM_TABLET)); - QObject* qmlSurface = tablet->getTabletSurface(); - - OffscreenQmlSurface* surface = dynamic_cast(qmlSurface); - - if (!surface) { + if (!tablet) { return nullptr; } - QQuickWindow* window = surface->getWindow(); - return window; + + auto* qmlSurface = tablet->getTabletSurface(); + if (!qmlSurface) { + return nullptr; + } + + return qmlSurface->getWindow(); } void TabletScriptingInterface::processMenuEvents(QObject* object, const QKeyEvent* event) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); switch (event->key()) { case Qt::Key_Down: QMetaObject::invokeMethod(object, "nextItem"); @@ -141,6 +124,7 @@ void TabletScriptingInterface::processMenuEvents(QObject* object, const QKeyEven } void TabletScriptingInterface::processTabletEvents(QObject* object, const QKeyEvent* event) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); switch (event->key()) { case Qt::Key_Down: QMetaObject::invokeMethod(object, "downItem"); @@ -167,8 +151,8 @@ void TabletScriptingInterface::processTabletEvents(QObject* object, const QKeyEv } } - void TabletScriptingInterface::processEvent(const QKeyEvent* event) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); TabletProxy* tablet = qobject_cast(getTablet(SYSTEM_TABLET)); QObject* qmlTablet = tablet->getQmlTablet(); QObject* qmlMenu = tablet->getQmlMenu(); @@ -180,8 +164,8 @@ void TabletScriptingInterface::processEvent(const QKeyEvent* event) { } } -QObject* TabletScriptingInterface::getFlags() -{ +QObject* TabletScriptingInterface::getFlags() { + Q_ASSERT(QThread::currentThread() == qApp->thread()); auto offscreenUi = DependencyManager::get(); return offscreenUi->getFlags(); } @@ -199,10 +183,23 @@ class TabletRootWindow : public QmlWindowClass { }; TabletProxy::TabletProxy(QObject* parent, QString name) : QObject(parent), _name(name) { + if (QThread::currentThread() != qApp->thread()) { + qCWarning(uiLogging) << "Creating tablet proxy on wrong thread " << _name; + } +} + +TabletProxy::~TabletProxy() { + qCDebug(uiLogging) << "Destroying tablet proxy " << _name; + if (QThread::currentThread() != thread()) { + qCWarning(uiLogging) << "Destroying tablet proxy on wrong thread" << _name; + } } void TabletProxy::setToolbarMode(bool toolbarMode) { - std::lock_guard guard(_tabletMutex); + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "setToolbarMode", Q_ARG(bool, toolbarMode)); + return; + } if (toolbarMode == _toolbarMode) { return; @@ -245,25 +242,23 @@ void TabletProxy::setToolbarMode(bool toolbarMode) { } static void addButtonProxyToQmlTablet(QQuickItem* qmlTablet, TabletButtonProxy* buttonProxy) { - QVariant resultVar; - Qt::ConnectionType connectionType = Qt::AutoConnection; - if (QThread::currentThread() != qmlTablet->thread()) { - connectionType = Qt::BlockingQueuedConnection; + Q_ASSERT(QThread::currentThread() == qApp->thread()); + if (buttonProxy == NULL){ + qCCritical(uiLogging) << "TabletScriptingInterface addButtonProxyToQmlTablet buttonProxy is NULL"; + return; } - if (buttonProxy == NULL){ - qCCritical(scriptengine) << "TabletScriptingInterface addButtonProxyToQmlTablet buttonProxy is NULL"; - return; - } - bool hasResult = QMetaObject::invokeMethod(qmlTablet, "addButtonProxy", connectionType, + + QVariant resultVar; + bool hasResult = QMetaObject::invokeMethod(qmlTablet, "addButtonProxy", Qt::DirectConnection, Q_RETURN_ARG(QVariant, resultVar), Q_ARG(QVariant, buttonProxy->getProperties())); if (!hasResult) { - qCWarning(scriptengine) << "TabletScriptingInterface addButtonProxyToQmlTablet has no result"; + qCWarning(uiLogging) << "TabletScriptingInterface addButtonProxyToQmlTablet has no result"; return; } QObject* qmlButton = qvariant_cast(resultVar); if (!qmlButton) { - qCWarning(scriptengine) << "TabletScriptingInterface addButtonProxyToQmlTablet result not a QObject"; + qCWarning(uiLogging) << "TabletScriptingInterface addButtonProxyToQmlTablet result not a QObject"; return; } QObject::connect(qmlButton, SIGNAL(clicked()), buttonProxy, SLOT(clickedSlot())); @@ -281,6 +276,11 @@ static QString getUsername() { } void TabletProxy::initialScreen(const QVariant& url) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "initialScreen", Q_ARG(QVariant, url)); + return; + } + if (_qmlTabletRoot) { pushOntoStack(url); } else { @@ -290,34 +290,49 @@ void TabletProxy::initialScreen(const QVariant& url) { } bool TabletProxy::isMessageDialogOpen() { - if (_qmlTabletRoot) { - QVariant result; - QMetaObject::invokeMethod(_qmlTabletRoot, "isDialogOpen",Qt::DirectConnection, - Q_RETURN_ARG(QVariant, result)); - - return result.toBool(); + if (QThread::currentThread() != thread()) { + bool result = false; + QMetaObject::invokeMethod(this, "isMessageDialogOpen", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, result)); + return result; } - return false; + if (!_qmlTabletRoot) { + return false; + } + + QVariant result; + QMetaObject::invokeMethod(_qmlTabletRoot, "isDialogOpen",Qt::DirectConnection, + Q_RETURN_ARG(QVariant, result)); + return result.toBool(); } void TabletProxy::emitWebEvent(QVariant msg) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "emitWebEvent", Q_ARG(QVariant, msg)); + return; + } emit webEventReceived(msg); } bool TabletProxy::isPathLoaded(QVariant path) { + if (QThread::currentThread() != thread()) { + bool result = false; + QMetaObject::invokeMethod(this, "isPathLoaded", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, result), Q_ARG(QVariant, path)); + return result; + } + return path.toString() == _currentPathLoaded.toString(); } -void TabletProxy::setQmlTabletRoot(QQuickItem* qmlTabletRoot, QObject* qmlOffscreenSurface) { - std::lock_guard guard(_tabletMutex); - _qmlOffscreenSurface = qmlOffscreenSurface; - _qmlTabletRoot = qmlTabletRoot; - if (_qmlTabletRoot && _qmlOffscreenSurface) { - QObject::connect(_qmlOffscreenSurface, SIGNAL(webEventReceived(QVariant)), this, SLOT(emitWebEvent(QVariant)), Qt::DirectConnection); +void TabletProxy::setQmlTabletRoot(OffscreenQmlSurface* qmlOffscreenSurface) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); + _qmlOffscreenSurface = qmlOffscreenSurface; + _qmlTabletRoot = qmlOffscreenSurface ? qmlOffscreenSurface->getRootItem() : nullptr; + if (_qmlTabletRoot && _qmlOffscreenSurface) { + QObject::connect(_qmlOffscreenSurface, SIGNAL(webEventReceived(QVariant)), this, SLOT(emitWebEvent(QVariant))); // forward qml surface events to interface js - connect(dynamic_cast(_qmlOffscreenSurface), &OffscreenQmlSurface::fromQml, [this](QVariant message) { + connect(_qmlOffscreenSurface, &OffscreenQmlSurface::fromQml, [this](QVariant message) { if (message.canConvert()) { emit fromQml(qvariant_cast(message).toVariant()); } else if (message.canConvert()) { @@ -330,7 +345,7 @@ void TabletProxy::setQmlTabletRoot(QQuickItem* qmlTabletRoot, QObject* qmlOffscr if (_toolbarMode) { // if someone creates the tablet in toolbar mode, make sure to display the home screen on the tablet. auto loader = _qmlTabletRoot->findChild("loader"); - QObject::connect(loader, SIGNAL(loaded()), this, SLOT(addButtonsToHomeScreen()), Qt::DirectConnection); + QObject::connect(loader, SIGNAL(loaded()), this, SLOT(addButtonsToHomeScreen())); QMetaObject::invokeMethod(_qmlTabletRoot, "loadSource", Q_ARG(const QVariant&, QVariant(TABLET_SOURCE_URL))); } @@ -360,9 +375,18 @@ void TabletProxy::setQmlTabletRoot(QQuickItem* qmlTabletRoot, QObject* qmlOffscr } void TabletProxy::gotoHomeScreen() { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "gotoHomeScreen"); + return; + } loadHomeScreen(false); } + void TabletProxy::gotoMenuScreen(const QString& submenu) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "gotoMenuScreen", Q_ARG(QString, submenu)); + return; + } QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { @@ -385,6 +409,11 @@ void TabletProxy::gotoMenuScreen(const QString& submenu) { } void TabletProxy::loadQMLOnTop(const QVariant& path) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "loadQMLOnTop", Q_ARG(QVariant, path)); + return; + } + QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { root = _qmlTabletRoot; @@ -396,11 +425,16 @@ void TabletProxy::loadQMLOnTop(const QVariant& path) { QMetaObject::invokeMethod(root, "loadQMLOnTop", Q_ARG(const QVariant&, path)); QMetaObject::invokeMethod(root, "setShown", Q_ARG(const QVariant&, QVariant(true))); } else { - qCDebug(scriptengine) << "tablet cannot load QML because _qmlTabletRoot is null"; + qCDebug(uiLogging) << "tablet cannot load QML because _qmlTabletRoot is null"; } } void TabletProxy::returnToPreviousApp() { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "returnToPreviousApp"); + return; + } + QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { root = _qmlTabletRoot; @@ -412,11 +446,15 @@ void TabletProxy::returnToPreviousApp() { QMetaObject::invokeMethod(root, "returnToPreviousApp"); QMetaObject::invokeMethod(root, "setShown", Q_ARG(const QVariant&, QVariant(true))); } else { - qCDebug(scriptengine) << "tablet cannot load QML because _qmlTabletRoot is null"; + qCDebug(uiLogging) << "tablet cannot load QML because _qmlTabletRoot is null"; } } void TabletProxy::loadQMLSource(const QVariant& path) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "loadQMLSource", Q_ARG(QVariant, path)); + return; + } QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { @@ -435,11 +473,17 @@ void TabletProxy::loadQMLSource(const QVariant& path) { QMetaObject::invokeMethod(root, "setShown", Q_ARG(const QVariant&, QVariant(true))); } } else { - qCDebug(scriptengine) << "tablet cannot load QML because _qmlTabletRoot is null"; + qCDebug(uiLogging) << "tablet cannot load QML because _qmlTabletRoot is null"; } } bool TabletProxy::pushOntoStack(const QVariant& path) { + if (QThread::currentThread() != thread()) { + bool result = false; + QMetaObject::invokeMethod(this, "pushOntoStack", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, result), Q_ARG(QVariant, path)); + return result; + } + QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { root = _qmlTabletRoot; @@ -455,13 +499,18 @@ bool TabletProxy::pushOntoStack(const QVariant& path) { loadQMLSource(path); } } else { - qCDebug(scriptengine) << "tablet cannot push QML because _qmlTabletRoot or _desktopWindow is null"; + qCDebug(uiLogging) << "tablet cannot push QML because _qmlTabletRoot or _desktopWindow is null"; } return root; } void TabletProxy::popFromStack() { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "popFromStack"); + return; + } + QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { root = _qmlTabletRoot; @@ -473,15 +522,20 @@ void TabletProxy::popFromStack() { auto stack = root->findChild("stack"); QMetaObject::invokeMethod(stack, "popSource"); } else { - qCDebug(scriptengine) << "tablet cannot pop QML because _qmlTabletRoot or _desktopWindow is null"; + qCDebug(uiLogging) << "tablet cannot pop QML because _qmlTabletRoot or _desktopWindow is null"; } } void TabletProxy::loadHomeScreen(bool forceOntoHomeScreen) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "loadHomeScreen", Q_ARG(bool, forceOntoHomeScreen)); + return; + } + if ((_state != State::Home && _state != State::Uninitialized) || forceOntoHomeScreen) { if (!_toolbarMode && _qmlTabletRoot) { auto loader = _qmlTabletRoot->findChild("loader"); - QObject::connect(loader, SIGNAL(loaded()), this, SLOT(addButtonsToHomeScreen()), Qt::DirectConnection); + QObject::connect(loader, SIGNAL(loaded()), this, SLOT(addButtonsToHomeScreen())); QMetaObject::invokeMethod(_qmlTabletRoot, "loadSource", Q_ARG(const QVariant&, QVariant(TABLET_SOURCE_URL))); QMetaObject::invokeMethod(_qmlTabletRoot, "playButtonClickSound"); } else if (_toolbarMode && _desktopWindow) { @@ -505,6 +559,11 @@ void TabletProxy::loadWebScreenOnTop(const QVariant& url) { } void TabletProxy::loadWebScreenOnTop(const QVariant& url, const QString& injectJavaScriptUrl) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "loadWebScreenOnTop", Q_ARG(QVariant, url), Q_ARG(QString, injectJavaScriptUrl)); + return; + } + QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { root = _qmlTabletRoot; @@ -521,6 +580,10 @@ void TabletProxy::loadWebScreenOnTop(const QVariant& url, const QString& injectJ } void TabletProxy::gotoWebScreen(const QString& url, const QString& injectedJavaScriptUrl) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "gotoWebScreen", Q_ARG(QString, url), Q_ARG(QString, injectedJavaScriptUrl)); + return; + } QObject* root = nullptr; if (!_toolbarMode && _qmlTabletRoot) { @@ -540,57 +603,59 @@ void TabletProxy::gotoWebScreen(const QString& url, const QString& injectedJavaS _currentPathLoaded = QVariant(url); } -QObject* TabletProxy::addButton(const QVariant& properties) { +TabletButtonProxy* TabletProxy::addButton(const QVariant& properties) { + if (QThread::currentThread() != thread()) { + TabletButtonProxy* result = nullptr; + QMetaObject::invokeMethod(this, "addButton", Qt::BlockingQueuedConnection, Q_RETURN_ARG(TabletButtonProxy*, result), Q_ARG(QVariant, properties)); + return result; + } + auto tabletButtonProxy = QSharedPointer(new TabletButtonProxy(properties.toMap())); - std::unique_lock guard(_tabletMutex); _tabletButtonProxies.push_back(tabletButtonProxy); if (!_toolbarMode && _qmlTabletRoot) { auto tablet = getQmlTablet(); if (tablet) { addButtonProxyToQmlTablet(tablet, tabletButtonProxy.data()); } else { - qCCritical(scriptengine) << "Could not find tablet in TabletRoot.qml"; + qCCritical(uiLogging) << "Could not find tablet in TabletRoot.qml"; } } else if (_toolbarMode) { auto tabletScriptingInterface = DependencyManager::get(); - QObject* toolbarProxy = tabletScriptingInterface->getSystemToolbarProxy(); - - Qt::ConnectionType connectionType = Qt::AutoConnection; - if (QThread::currentThread() != toolbarProxy->thread()) { - connectionType = Qt::BlockingQueuedConnection; - } - - guard.unlock(); - - // copy properties from tablet button proxy to toolbar button proxy. - QObject* toolbarButtonProxy = nullptr; - bool hasResult = QMetaObject::invokeMethod(toolbarProxy, "addButton", connectionType, Q_RETURN_ARG(QObject*, toolbarButtonProxy), Q_ARG(QVariant, tabletButtonProxy->getProperties())); - if (hasResult) { - tabletButtonProxy->setToolbarButtonProxy(toolbarButtonProxy); - } else { - qCWarning(scriptengine) << "ToolbarProxy addButton has no result"; + auto toolbarProxy = tabletScriptingInterface->getSystemToolbarProxy(); + if (toolbarProxy) { + // copy properties from tablet button proxy to toolbar button proxy. + toolbarProxy->addButton(tabletButtonProxy->getProperties()); } } return tabletButtonProxy.data(); } bool TabletProxy::onHomeScreen() { + if (QThread::currentThread() != thread()) { + bool result = false; + QMetaObject::invokeMethod(this, "onHomeScreen", Qt::BlockingQueuedConnection, Q_RETURN_ARG(bool, result)); + return result; + } + return _state == State::Home; } -void TabletProxy::removeButton(QObject* tabletButtonProxy) { - std::unique_lock guard(_tabletMutex); +void TabletProxy::removeButton(TabletButtonProxy* tabletButtonProxy) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "removeButton", Q_ARG(TabletButtonProxy*, tabletButtonProxy)); + return; + } auto tablet = getQmlTablet(); if (!tablet) { - qCCritical(scriptengine) << "Could not find tablet in TabletRoot.qml"; + qCCritical(uiLogging) << "Could not find tablet in TabletRoot.qml"; } QSharedPointer buttonProxy; { auto iter = std::find(_tabletButtonProxies.begin(), _tabletButtonProxies.end(), tabletButtonProxy); if (iter == _tabletButtonProxies.end()) { - qCWarning(scriptengine) << "TabletProxy::removeButton() could not find button " << tabletButtonProxy; + qCWarning(uiLogging) << "TabletProxy::removeButton() could not find button " << tabletButtonProxy; return; } buttonProxy = *iter; @@ -600,21 +665,23 @@ void TabletProxy::removeButton(QObject* tabletButtonProxy) { if (!_toolbarMode && _qmlTabletRoot) { buttonProxy->setQmlButton(nullptr); if (tablet) { - guard.unlock(); QMetaObject::invokeMethod(tablet, "removeButtonProxy", Qt::AutoConnection, Q_ARG(QVariant, buttonProxy->getProperties())); } } else if (_toolbarMode) { auto tabletScriptingInterface = DependencyManager::get(); QObject* toolbarProxy = tabletScriptingInterface->getSystemToolbarProxy(); - // remove button from toolbarProxy - guard.unlock(); QMetaObject::invokeMethod(toolbarProxy, "removeButton", Qt::AutoConnection, Q_ARG(QVariant, buttonProxy->getUuid().toString())); buttonProxy->setToolbarButtonProxy(nullptr); } } void TabletProxy::emitScriptEvent(QVariant msg) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "emitScriptEvent", Q_ARG(QVariant, msg)); + return; + } + if (!_toolbarMode && _qmlOffscreenSurface) { QMetaObject::invokeMethod(_qmlOffscreenSurface, "emitScriptEvent", Qt::AutoConnection, Q_ARG(QVariant, msg)); } else if (_toolbarMode && _desktopWindow) { @@ -623,6 +690,11 @@ void TabletProxy::emitScriptEvent(QVariant msg) { } void TabletProxy::sendToQml(QVariant msg) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "sendToQml", Q_ARG(QVariant, msg)); + return; + } + if (!_toolbarMode && _qmlOffscreenSurface) { QMetaObject::invokeMethod(_qmlOffscreenSurface, "sendToQml", Qt::AutoConnection, Q_ARG(QVariant, msg)); } else if (_toolbarMode && _desktopWindow) { @@ -644,7 +716,7 @@ void TabletProxy::addButtonsToHomeScreen() { QObject::disconnect(loader, SIGNAL(loaded()), this, SLOT(addButtonsToHomeScreen())); } -QObject* TabletProxy::getTabletSurface() { +OffscreenQmlSurface* TabletProxy::getTabletSurface() { return _qmlOffscreenSurface; } @@ -678,7 +750,7 @@ void TabletProxy::addButtonsToToolbar() { if (hasResult) { buttonProxy->setToolbarButtonProxy(toolbarButtonProxy); } else { - qCWarning(scriptengine) << "ToolbarProxy addButton has no result"; + qCWarning(uiLogging) << "ToolbarProxy addButton has no result"; } } @@ -753,34 +825,55 @@ TabletButtonProxy::TabletButtonProxy(const QVariantMap& properties) : _properties[UUID_KEY] = _uuid; _properties[OBJECT_NAME_KEY] = _uuid.toString(); _properties[STABLE_ORDER_KEY] = _stableOrder; + if (QThread::currentThread() != qApp->thread()) { + qCWarning(uiLogging) << "Creating tablet button proxy on wrong thread"; + } +} + +TabletButtonProxy::~TabletButtonProxy() { + qCDebug(uiLogging) << "Destroying tablet button proxy " ; + if (QThread::currentThread() != thread()) { + qCWarning(uiLogging) << "Destroying tablet button proxy on wrong thread"; + } } void TabletButtonProxy::setQmlButton(QQuickItem* qmlButton) { - std::lock_guard guard(_buttonMutex); + Q_ASSERT(QThread::currentThread() == qApp->thread()); _qmlButton = qmlButton; } void TabletButtonProxy::setToolbarButtonProxy(QObject* toolbarButtonProxy) { - std::lock_guard guard(_buttonMutex); _toolbarButtonProxy = toolbarButtonProxy; if (_toolbarButtonProxy) { QObject::connect(_toolbarButtonProxy, SIGNAL(clicked()), this, SLOT(clickedSlot())); } } -QVariantMap TabletButtonProxy::getProperties() const { - std::lock_guard guard(_buttonMutex); +QVariantMap TabletButtonProxy::getProperties() { + if (QThread::currentThread() != thread()) { + QVariantMap result; + QMetaObject::invokeMethod(this, "getProperties", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QVariantMap, result)); + return result; + } + return _properties; } void TabletButtonProxy::editProperties(QVariantMap properties) { - std::lock_guard guard(_buttonMutex); + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "editProperties", Qt::BlockingQueuedConnection, Q_ARG(QVariantMap, properties)); + return; + } QVariantMap::const_iterator iter = properties.constBegin(); while (iter != properties.constEnd()) { - _properties[iter.key()] = iter.value(); - if (_qmlButton) { - QMetaObject::invokeMethod(_qmlButton, "changeProperty", Qt::AutoConnection, Q_ARG(QVariant, QVariant(iter.key())), Q_ARG(QVariant, iter.value())); + const auto& key = iter.key(); + const auto& value = iter.value(); + if (!_properties.contains(key) || _properties[key] != value) { + _properties[iter.key()] = iter.value(); + if (_qmlButton) { + QMetaObject::invokeMethod(_qmlButton, "changeProperty", Qt::AutoConnection, Q_ARG(QVariant, QVariant(iter.key())), Q_ARG(QVariant, iter.value())); + } } ++iter; } @@ -789,5 +882,3 @@ void TabletButtonProxy::editProperties(QVariantMap properties) { QMetaObject::invokeMethod(_toolbarButtonProxy, "editProperties", Qt::AutoConnection, Q_ARG(QVariantMap, properties)); } } - -#include "TabletScriptingInterface.moc" diff --git a/libraries/script-engine/src/TabletScriptingInterface.h b/libraries/ui/src/ui/TabletScriptingInterface.h similarity index 90% rename from libraries/script-engine/src/TabletScriptingInterface.h rename to libraries/ui/src/ui/TabletScriptingInterface.h index 23d7ecaea4..0303a0a1c4 100644 --- a/libraries/script-engine/src/TabletScriptingInterface.h +++ b/libraries/ui/src/ui/TabletScriptingInterface.h @@ -26,9 +26,13 @@ #include +class ToolbarProxy; +class ToolbarScriptingInterface; + class TabletProxy; class TabletButtonProxy; class QmlWindowClass; +class OffscreenQmlSurface; /**jsdoc * @namespace Tablet @@ -37,9 +41,9 @@ class TabletScriptingInterface : public QObject, public Dependency { Q_OBJECT public: TabletScriptingInterface(); + ~TabletScriptingInterface(); - void setToolbarScriptingInterface(QObject* toolbarScriptingInterface) { _toolbarScriptingInterface = toolbarScriptingInterface; } - QObject* getSystemToolbarProxy(); + void setToolbarScriptingInterface(ToolbarScriptingInterface* toolbarScriptingInterface) { _toolbarScriptingInterface = toolbarScriptingInterface; } /**jsdoc * Creates or retruns a new TabletProxy and returns it. @@ -51,7 +55,7 @@ public: void setToolbarMode(bool toolbarMode); - void setQmlTabletRoot(QString tabletId, QQuickItem* qmlTabletRoot, QObject* qmlOffscreenSurface); + void setQmlTabletRoot(QString tabletId, OffscreenQmlSurface* offscreenQmlSurface); void processEvent(const QKeyEvent* event); @@ -67,13 +71,14 @@ signals: void tabletNotification(); private: + friend class TabletProxy; void processMenuEvents(QObject* object, const QKeyEvent* event); void processTabletEvents(QObject* object, const QKeyEvent* event); + ToolbarProxy* getSystemToolbarProxy(); protected: - std::mutex _mapMutex; std::map _tabletProxies; - QObject* _toolbarScriptingInterface { nullptr }; + ToolbarScriptingInterface* _toolbarScriptingInterface { nullptr }; bool _toolbarMode { false }; }; @@ -91,18 +96,19 @@ class TabletProxy : public QObject { Q_PROPERTY(bool tabletShown MEMBER _tabletShown NOTIFY tabletShownChanged) public: TabletProxy(QObject* parent, QString name); + ~TabletProxy(); - void setQmlTabletRoot(QQuickItem* qmlTabletRoot, QObject* qmlOffscreenSurface); - - Q_INVOKABLE void gotoMenuScreen(const QString& submenu = ""); - - QString getName() const { return _name; } + void setQmlTabletRoot(OffscreenQmlSurface* offscreenQmlSurface); + const QString getName() const { return _name; } bool getToolbarMode() const { return _toolbarMode; } void setToolbarMode(bool toolbarMode); + + Q_INVOKABLE void gotoMenuScreen(const QString& submenu = ""); Q_INVOKABLE void initialScreen(const QVariant& url); + /**jsdoc * transition to the home screen * @function TabletProxy#gotoHomeScreen @@ -143,14 +149,14 @@ public: * @param properties {Object} button properties UI_TABLET_HACK: enumerate these when we figure out what they should be! * @returns {TabletButtonProxy} */ - Q_INVOKABLE QObject* addButton(const QVariant& properties); + Q_INVOKABLE TabletButtonProxy* addButton(const QVariant& properties); /**jsdoc * removes button from the tablet * @function TabletProxy.removeButton * @param tabletButtonProxy {TabletButtonProxy} button to be removed */ - Q_INVOKABLE void removeButton(QObject* tabletButtonProxy); + Q_INVOKABLE void removeButton(TabletButtonProxy* tabletButtonProxy); /**jsdoc * Used to send an event to the html/js embedded in the tablet @@ -184,7 +190,7 @@ public: QQuickItem* getTabletRoot() const { return _qmlTabletRoot; } - QObject* getTabletSurface(); + OffscreenQmlSurface* getTabletSurface(); QQuickItem* getQmlTablet() const; @@ -236,10 +242,9 @@ protected: QVariant _initialPath { "" }; QVariant _currentPathLoaded { "" }; QString _name; - std::mutex _tabletMutex; std::vector> _tabletButtonProxies; QQuickItem* _qmlTabletRoot { nullptr }; - QObject* _qmlOffscreenSurface { nullptr }; + OffscreenQmlSurface* _qmlOffscreenSurface { nullptr }; QmlWindowClass* _desktopWindow { nullptr }; bool _toolbarMode { false }; bool _tabletShown { false }; @@ -251,9 +256,6 @@ protected: Q_DECLARE_METATYPE(TabletProxy*); -QScriptValue tabletToScriptValue(QScriptEngine* engine, TabletProxy* const &in); -void tabletFromScriptValue(const QScriptValue& value, TabletProxy* &out); - /**jsdoc * @class TabletButtonProxy * @property uuid {QUuid} READ_ONLY: uniquely identifies this button @@ -263,6 +265,7 @@ class TabletButtonProxy : public QObject { Q_PROPERTY(QUuid uuid READ getUuid) public: TabletButtonProxy(const QVariantMap& properties); + ~TabletButtonProxy(); void setQmlButton(QQuickItem* qmlButton); void setToolbarButtonProxy(QObject* toolbarButtonProxy); @@ -274,7 +277,7 @@ public: * @function TabletButtonProxy#getProperties * @returns {ButtonProperties} */ - Q_INVOKABLE QVariantMap getProperties() const; + Q_INVOKABLE QVariantMap getProperties(); /**jsdoc * Replace the values of some of this button's properties @@ -297,12 +300,13 @@ signals: protected: QUuid _uuid; int _stableOrder; - mutable std::mutex _buttonMutex; QQuickItem* _qmlButton { nullptr }; QObject* _toolbarButtonProxy { nullptr }; QVariantMap _properties; }; +Q_DECLARE_METATYPE(TabletButtonProxy*); + /**jsdoc * @typedef TabletButtonProxy.ButtonProperties * @property {string} icon - url to button icon. (50 x 50) diff --git a/libraries/ui/src/ui/ToolbarScriptingInterface.cpp b/libraries/ui/src/ui/ToolbarScriptingInterface.cpp new file mode 100644 index 0000000000..330c652cdc --- /dev/null +++ b/libraries/ui/src/ui/ToolbarScriptingInterface.cpp @@ -0,0 +1,124 @@ +// +// Created by Bradley Austin Davis on 2016-06-16 +// Copyright 2013-2016 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#include "ToolbarScriptingInterface.h" + +#include +#include +#include +#include +#include "../OffscreenUi.h" + +QScriptValue toolbarToScriptValue(QScriptEngine* engine, ToolbarProxy* const &in) { + if (!in) { + return engine->undefinedValue(); + } + return engine->newQObject(in, QScriptEngine::QtOwnership, QScriptEngine::ExcludeDeleteLater | QScriptEngine::ExcludeChildObjects); +} + +void toolbarFromScriptValue(const QScriptValue& value, ToolbarProxy* &out) { + out = qobject_cast(value.toQObject()); +} + +QScriptValue toolbarButtonToScriptValue(QScriptEngine* engine, ToolbarButtonProxy* const &in) { + if (!in) { + return engine->undefinedValue(); + } + return engine->newQObject(in, QScriptEngine::QtOwnership, QScriptEngine::ExcludeDeleteLater | QScriptEngine::ExcludeChildObjects); +} + +void toolbarButtonFromScriptValue(const QScriptValue& value, ToolbarButtonProxy* &out) { + out = qobject_cast(value.toQObject()); +} + + +ToolbarButtonProxy::ToolbarButtonProxy(QObject* qmlObject, QObject* parent) : QmlWrapper(qmlObject, parent) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); + _qmlButton = qobject_cast(qmlObject); + connect(qmlObject, SIGNAL(clicked()), this, SIGNAL(clicked())); +} + +void ToolbarButtonProxy::editProperties(const QVariantMap& properties) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "editProperties", Q_ARG(QVariantMap, properties)); + return; + } + + QVariantMap::const_iterator iter = properties.constBegin(); + while (iter != properties.constEnd()) { + _properties[iter.key()] = iter.value(); + if (_qmlButton) { + // [01/25 14:26:20] [WARNING] [default] QMetaObject::invokeMethod: No such method ToolbarButton_QMLTYPE_195::changeProperty(QVariant,QVariant) + QMetaObject::invokeMethod(_qmlButton, "changeProperty", Qt::AutoConnection, + Q_ARG(QVariant, QVariant(iter.key())), Q_ARG(QVariant, iter.value())); + } + ++iter; + } +} + +ToolbarProxy::ToolbarProxy(QObject* qmlObject, QObject* parent) : QmlWrapper(qmlObject, parent) { + Q_ASSERT(QThread::currentThread() == qApp->thread()); +} + +ToolbarButtonProxy* ToolbarProxy::addButton(const QVariant& properties) { + if (QThread::currentThread() != thread()) { + ToolbarButtonProxy* result = nullptr; + QMetaObject::invokeMethod(this, "addButton", Qt::BlockingQueuedConnection, Q_RETURN_ARG(ToolbarButtonProxy*, result), Q_ARG(QVariant, properties)); + return result; + } + + QVariant resultVar; + bool invokeResult = QMetaObject::invokeMethod(_qmlObject, "addButton", Q_RETURN_ARG(QVariant, resultVar), Q_ARG(QVariant, properties)); + if (!invokeResult) { + return nullptr; + } + + QObject* rawButton = qvariant_cast(resultVar); + if (!rawButton) { + return nullptr; + } + + return new ToolbarButtonProxy(rawButton, this); +} + +void ToolbarProxy::removeButton(const QVariant& name) { + if (QThread::currentThread() != thread()) { + QMetaObject::invokeMethod(this, "removeButton", Q_ARG(QVariant, name)); + return; + } + + QMetaObject::invokeMethod(_qmlObject, "removeButton", Q_ARG(QVariant, name)); +} + + +ToolbarProxy* ToolbarScriptingInterface::getToolbar(const QString& toolbarId) { + if (QThread::currentThread() != thread()) { + ToolbarProxy* result = nullptr; + QMetaObject::invokeMethod(this, "getToolbar", Qt::BlockingQueuedConnection, Q_RETURN_ARG(ToolbarProxy*, result), Q_ARG(QString, toolbarId)); + return result; + } + + auto offscreenUi = DependencyManager::get(); + auto desktop = offscreenUi->getDesktop(); + Qt::ConnectionType connectionType = Qt::AutoConnection; + if (QThread::currentThread() != desktop->thread()) { + connectionType = Qt::BlockingQueuedConnection; + } + QVariant resultVar; + bool invokeResult = QMetaObject::invokeMethod(desktop, "getToolbar", connectionType, Q_RETURN_ARG(QVariant, resultVar), Q_ARG(QVariant, toolbarId)); + if (!invokeResult) { + return nullptr; + } + + QObject* rawToolbar = qvariant_cast(resultVar); + if (!rawToolbar) { + return nullptr; + } + + return new ToolbarProxy(rawToolbar); +} diff --git a/libraries/ui/src/ui/ToolbarScriptingInterface.h b/libraries/ui/src/ui/ToolbarScriptingInterface.h new file mode 100644 index 0000000000..108cf6bdd5 --- /dev/null +++ b/libraries/ui/src/ui/ToolbarScriptingInterface.h @@ -0,0 +1,56 @@ +// +// Created by Bradley Austin Davis on 2016-06-16 +// Copyright 2013-2016 High Fidelity, Inc. +// +// Distributed under the Apache License, Version 2.0. +// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html +// + +#ifndef hifi_ToolbarScriptingInterface_h +#define hifi_ToolbarScriptingInterface_h + +#include + +#include +#include + +#include +#include "QmlWrapper.h" + +class QQuickItem; + +class ToolbarButtonProxy : public QmlWrapper { + Q_OBJECT + +public: + ToolbarButtonProxy(QObject* qmlObject, QObject* parent = nullptr); + Q_INVOKABLE void editProperties(const QVariantMap& properties); + +signals: + void clicked(); + +protected: + QQuickItem* _qmlButton { nullptr }; + QVariantMap _properties; +}; + +Q_DECLARE_METATYPE(ToolbarButtonProxy*); + +class ToolbarProxy : public QmlWrapper { + Q_OBJECT +public: + ToolbarProxy(QObject* qmlObject, QObject* parent = nullptr); + Q_INVOKABLE ToolbarButtonProxy* addButton(const QVariant& properties); + Q_INVOKABLE void removeButton(const QVariant& name); +}; + +Q_DECLARE_METATYPE(ToolbarProxy*); + +class ToolbarScriptingInterface : public QObject, public Dependency { + Q_OBJECT +public: + Q_INVOKABLE ToolbarProxy* getToolbar(const QString& toolbarId); +}; + + +#endif // hifi_ToolbarScriptingInterface_h diff --git a/libraries/networking/src/FileTypeProfile.cpp b/libraries/ui/src/ui/types/FileTypeProfile.cpp similarity index 100% rename from libraries/networking/src/FileTypeProfile.cpp rename to libraries/ui/src/ui/types/FileTypeProfile.cpp diff --git a/libraries/networking/src/FileTypeProfile.h b/libraries/ui/src/ui/types/FileTypeProfile.h similarity index 100% rename from libraries/networking/src/FileTypeProfile.h rename to libraries/ui/src/ui/types/FileTypeProfile.h diff --git a/libraries/networking/src/FileTypeRequestInterceptor.cpp b/libraries/ui/src/ui/types/FileTypeRequestInterceptor.cpp similarity index 100% rename from libraries/networking/src/FileTypeRequestInterceptor.cpp rename to libraries/ui/src/ui/types/FileTypeRequestInterceptor.cpp diff --git a/libraries/networking/src/FileTypeRequestInterceptor.h b/libraries/ui/src/ui/types/FileTypeRequestInterceptor.h similarity index 100% rename from libraries/networking/src/FileTypeRequestInterceptor.h rename to libraries/ui/src/ui/types/FileTypeRequestInterceptor.h diff --git a/libraries/networking/src/HFTabletWebEngineProfile.cpp b/libraries/ui/src/ui/types/HFTabletWebEngineProfile.cpp similarity index 100% rename from libraries/networking/src/HFTabletWebEngineProfile.cpp rename to libraries/ui/src/ui/types/HFTabletWebEngineProfile.cpp diff --git a/libraries/networking/src/HFTabletWebEngineProfile.h b/libraries/ui/src/ui/types/HFTabletWebEngineProfile.h similarity index 100% rename from libraries/networking/src/HFTabletWebEngineProfile.h rename to libraries/ui/src/ui/types/HFTabletWebEngineProfile.h diff --git a/libraries/networking/src/HFTabletWebEngineRequestInterceptor.cpp b/libraries/ui/src/ui/types/HFTabletWebEngineRequestInterceptor.cpp similarity index 100% rename from libraries/networking/src/HFTabletWebEngineRequestInterceptor.cpp rename to libraries/ui/src/ui/types/HFTabletWebEngineRequestInterceptor.cpp diff --git a/libraries/networking/src/HFTabletWebEngineRequestInterceptor.h b/libraries/ui/src/ui/types/HFTabletWebEngineRequestInterceptor.h similarity index 100% rename from libraries/networking/src/HFTabletWebEngineRequestInterceptor.h rename to libraries/ui/src/ui/types/HFTabletWebEngineRequestInterceptor.h diff --git a/libraries/networking/src/HFWebEngineProfile.cpp b/libraries/ui/src/ui/types/HFWebEngineProfile.cpp similarity index 100% rename from libraries/networking/src/HFWebEngineProfile.cpp rename to libraries/ui/src/ui/types/HFWebEngineProfile.cpp diff --git a/libraries/networking/src/HFWebEngineProfile.h b/libraries/ui/src/ui/types/HFWebEngineProfile.h similarity index 100% rename from libraries/networking/src/HFWebEngineProfile.h rename to libraries/ui/src/ui/types/HFWebEngineProfile.h diff --git a/libraries/networking/src/HFWebEngineRequestInterceptor.cpp b/libraries/ui/src/ui/types/HFWebEngineRequestInterceptor.cpp similarity index 100% rename from libraries/networking/src/HFWebEngineRequestInterceptor.cpp rename to libraries/ui/src/ui/types/HFWebEngineRequestInterceptor.cpp diff --git a/libraries/networking/src/HFWebEngineRequestInterceptor.h b/libraries/ui/src/ui/types/HFWebEngineRequestInterceptor.h similarity index 100% rename from libraries/networking/src/HFWebEngineRequestInterceptor.h rename to libraries/ui/src/ui/types/HFWebEngineRequestInterceptor.h diff --git a/libraries/networking/src/RequestFilters.cpp b/libraries/ui/src/ui/types/RequestFilters.cpp similarity index 100% rename from libraries/networking/src/RequestFilters.cpp rename to libraries/ui/src/ui/types/RequestFilters.cpp diff --git a/libraries/networking/src/RequestFilters.h b/libraries/ui/src/ui/types/RequestFilters.h similarity index 100% rename from libraries/networking/src/RequestFilters.h rename to libraries/ui/src/ui/types/RequestFilters.h diff --git a/libraries/script-engine/src/SoundEffect.cpp b/libraries/ui/src/ui/types/SoundEffect.cpp similarity index 97% rename from libraries/script-engine/src/SoundEffect.cpp rename to libraries/ui/src/ui/types/SoundEffect.cpp index bfc0ad2100..e35d009ef6 100644 --- a/libraries/script-engine/src/SoundEffect.cpp +++ b/libraries/ui/src/ui/types/SoundEffect.cpp @@ -41,5 +41,3 @@ void SoundEffect::play(QVariant position) { _injector = AudioInjector::playSound(samples, options); } } - -#include "SoundEffect.moc" diff --git a/libraries/script-engine/src/SoundEffect.h b/libraries/ui/src/ui/types/SoundEffect.h similarity index 100% rename from libraries/script-engine/src/SoundEffect.h rename to libraries/ui/src/ui/types/SoundEffect.h