// // Created by Bradley Austin Davis on 2015-12-15 // Copyright 2015 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 "QmlWindowClass.h" #include #include #include #include #include #include #include #include #include #include #include #include "OffscreenUi.h" static const char* const SOURCE_PROPERTY = "source"; static const char* const TITLE_PROPERTY = "title"; static const char* const EVENT_BRIDGE_PROPERTY = "eventBridge"; static const char* const WIDTH_PROPERTY = "width"; static const char* const HEIGHT_PROPERTY = "height"; static const char* const VISIBILE_PROPERTY = "visible"; static const char* const TOOLWINDOW_PROPERTY = "toolWindow"; static const uvec2 MAX_QML_WINDOW_SIZE { 1280, 720 }; static const uvec2 MIN_QML_WINDOW_SIZE { 120, 80 }; QVariantMap QmlWindowClass::parseArguments(QScriptContext* context) { const auto argumentCount = context->argumentCount(); QString title; QVariantMap properties; if (argumentCount > 1) { if (!context->argument(0).isUndefined()) { properties[TITLE_PROPERTY] = context->argument(0).toString(); } if (!context->argument(1).isUndefined()) { properties[SOURCE_PROPERTY] = context->argument(1).toString(); } if (context->argument(2).isNumber()) { properties[WIDTH_PROPERTY] = context->argument(2).toInt32(); } if (context->argument(3).isNumber()) { properties[HEIGHT_PROPERTY] = context->argument(3).toInt32(); } if (context->argument(4).isBool()) { properties[TOOLWINDOW_PROPERTY] = context->argument(4).toBool(); } } else { properties = context->argument(0).toVariant().toMap(); } QUrl url { properties[SOURCE_PROPERTY].toString() }; if (url.scheme() != "http" && url.scheme() != "https" && url.scheme() != "file" && url.scheme() != "about" && url.scheme() != "atp") { properties[SOURCE_PROPERTY] = QUrl::fromLocalFile(url.toString()).toString(); } return properties; } // Method called by Qt scripts to create a new web window in the overlay QScriptValue QmlWindowClass::constructor(QScriptContext* context, QScriptEngine* engine) { auto properties = parseArguments(context); QmlWindowClass* retVal { nullptr }; auto offscreenUi = DependencyManager::get(); offscreenUi->executeOnUiThread([&] { retVal = new QmlWindowClass(); retVal->initQml(properties); }, true); Q_ASSERT(retVal); connect(engine, &QScriptEngine::destroyed, retVal, &QmlWindowClass::deleteLater); return engine->newQObject(retVal); } QmlWindowClass::QmlWindowClass() { } void QmlWindowClass::initQml(QVariantMap properties) { auto offscreenUi = DependencyManager::get(); _toolWindow = properties.contains(TOOLWINDOW_PROPERTY) && properties[TOOLWINDOW_PROPERTY].toBool(); _source = properties[SOURCE_PROPERTY].toString(); if (_toolWindow) { // Build the event bridge and wrapper on the main thread _qmlWindow = offscreenUi->getToolWindow(); properties[EVENT_BRIDGE_PROPERTY] = QVariant::fromValue(this); QVariant newTabVar; bool invokeResult = QMetaObject::invokeMethod(_qmlWindow, "addWebTab", Qt::DirectConnection, Q_RETURN_ARG(QVariant, newTabVar), Q_ARG(QVariant, QVariant::fromValue(properties))); Q_ASSERT(invokeResult); } else { // Build the event bridge and wrapper on the main thread offscreenUi->load(qmlSource(), [&](QQmlContext* context, QObject* object) { _qmlWindow = object; _qmlWindow->setProperty("eventBridge", QVariant::fromValue(this)); context->engine()->setObjectOwnership(this, QQmlEngine::CppOwnership); context->engine()->setObjectOwnership(object, QQmlEngine::CppOwnership); if (properties.contains(TITLE_PROPERTY)) { object->setProperty(TITLE_PROPERTY, properties[TITLE_PROPERTY].toString()); } if (properties.contains(HEIGHT_PROPERTY) && properties.contains(WIDTH_PROPERTY)) { uvec2 requestedSize { properties[WIDTH_PROPERTY].toUInt(), properties[HEIGHT_PROPERTY].toUInt() }; requestedSize = glm::clamp(requestedSize, MIN_QML_WINDOW_SIZE, MAX_QML_WINDOW_SIZE); asQuickItem()->setSize(QSize(requestedSize.x, requestedSize.y)); } bool visible = !properties.contains(VISIBILE_PROPERTY) || properties[VISIBILE_PROPERTY].toBool(); object->setProperty(OFFSCREEN_VISIBILITY_PROPERTY, visible); object->setProperty(SOURCE_PROPERTY, _source); // Forward messages received from QML on to the script connect(_qmlWindow, SIGNAL(sendToScript(QVariant)), this, SLOT(qmlToScript(const QVariant&)), Qt::QueuedConnection); connect(_qmlWindow, SIGNAL(visibleChanged()), this, SIGNAL(visibleChanged()), Qt::QueuedConnection); connect(_qmlWindow, SIGNAL(resized(QSizeF)), this, SIGNAL(resized(QSizeF)), Qt::QueuedConnection); connect(_qmlWindow, SIGNAL(moved(QVector2D)), this, SLOT(hasMoved(QVector2D)), Qt::QueuedConnection); connect(_qmlWindow, SIGNAL(windowClosed()), this, SLOT(hasClosed()), Qt::QueuedConnection); }); } Q_ASSERT(_qmlWindow); Q_ASSERT(dynamic_cast(_qmlWindow.data())); } void QmlWindowClass::qmlToScript(const QVariant& message) { if (message.canConvert()) { emit fromQml(qvariant_cast(message).toVariant()); } else if (message.canConvert()) { emit fromQml(message.toString()); } else { qWarning() << "Unsupported message type " << message; } } void QmlWindowClass::sendToQml(const QVariant& message) { // Forward messages received from the script on to QML QMetaObject::invokeMethod(asQuickItem(), "fromScript", Qt::QueuedConnection, Q_ARG(QVariant, message)); } QmlWindowClass::~QmlWindowClass() { close(); } QQuickItem* QmlWindowClass::asQuickItem() const { if (_toolWindow) { return DependencyManager::get()->getToolWindow(); } return _qmlWindow.isNull() ? nullptr : dynamic_cast(_qmlWindow.data()); } void QmlWindowClass::setVisible(bool visible) { QQuickItem* targetWindow = asQuickItem(); if (_toolWindow) { // For tool window tabs we special case visibility as a function call on the tab parent // The tool window itself has special logic based on whether any tabs are visible QMetaObject::invokeMethod(targetWindow, "showTabForUrl", Qt::QueuedConnection, Q_ARG(QVariant, _source), Q_ARG(QVariant, visible)); } else { DependencyManager::get()->executeOnUiThread([=] { targetWindow->setProperty(OFFSCREEN_VISIBILITY_PROPERTY, visible); }); } } bool QmlWindowClass::isVisible() const { // The tool window itself has special logic based on whether any tabs are enabled return DependencyManager::get()->returnFromUiThread([&] { if (_qmlWindow.isNull()) { return QVariant::fromValue(false); } if (_toolWindow) { return QVariant::fromValue(dynamic_cast(_qmlWindow.data())->isEnabled()); } else { return QVariant::fromValue(asQuickItem()->isVisible()); } }).toBool(); } glm::vec2 QmlWindowClass::getPosition() const { QVariant result = DependencyManager::get()->returnFromUiThread([&]()->QVariant { if (_qmlWindow.isNull()) { return QVariant(QPointF(0, 0)); } return asQuickItem()->position(); }); return toGlm(result.toPointF()); } void QmlWindowClass::setPosition(const glm::vec2& position) { DependencyManager::get()->executeOnUiThread([=] { if (!_qmlWindow.isNull()) { asQuickItem()->setPosition(QPointF(position.x, position.y)); } }); } void QmlWindowClass::setPosition(int x, int y) { setPosition(glm::vec2(x, y)); } // FIXME move to GLM helpers glm::vec2 toGlm(const QSizeF& size) { return glm::vec2(size.width(), size.height()); } glm::vec2 QmlWindowClass::getSize() const { QVariant result = DependencyManager::get()->returnFromUiThread([&]()->QVariant { if (_qmlWindow.isNull()) { return QVariant(QSizeF(0, 0)); } QQuickItem* targetWindow = asQuickItem(); return QSizeF(targetWindow->width(), targetWindow->height()); }); return toGlm(result.toSizeF()); } void QmlWindowClass::setSize(const glm::vec2& size) { DependencyManager::get()->executeOnUiThread([=] { if (!_qmlWindow.isNull()) { asQuickItem()->setSize(QSizeF(size.x, size.y)); } }); } void QmlWindowClass::setSize(int width, int height) { setSize(glm::vec2(width, height)); } void QmlWindowClass::setTitle(const QString& title) { DependencyManager::get()->executeOnUiThread([=] { if (!_qmlWindow.isNull()) { asQuickItem()->setProperty(TITLE_PROPERTY, title); } }); } void QmlWindowClass::close() { if (_qmlWindow) { if (_toolWindow) { auto offscreenUi = DependencyManager::get(); offscreenUi->executeOnUiThread([=] { auto toolWindow = offscreenUi->getToolWindow(); auto invokeResult = QMetaObject::invokeMethod(toolWindow, "removeTabForUrl", Qt::DirectConnection, Q_ARG(QVariant, _source)); Q_ASSERT(invokeResult); }); } else { _qmlWindow->deleteLater(); } _qmlWindow = nullptr; } } void QmlWindowClass::hasMoved(QVector2D position) { emit moved(glm::vec2(position.x(), position.y())); } void QmlWindowClass::hasClosed() { emit closed(); } void QmlWindowClass::raise() { auto offscreenUi = DependencyManager::get(); offscreenUi->executeOnUiThread([=] { if (!_qmlWindow.isNull()) { QMetaObject::invokeMethod(asQuickItem(), "raise", Qt::DirectConnection); } }); }