// Copyright (C) 2021 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only #include "qquick3druntimeloader_p.h" #include #include #include #include #include #if QT_CONFIG(mimetype) #include #endif /*! \qmltype RuntimeLoader \inherits Node \inqmlmodule QtQuick3D.AssetUtils \since 6.2 \brief Imports a 3D asset at runtime. The RuntimeLoader type provides a way to load a 3D asset directly from source at runtime, without converting it to QtQuick3D's internal format first. RuntimeLoader supports .obj and glTF version 2.0 files in both in text (.gltf) and binary (.glb) formats. */ /*! \qmlproperty url RuntimeLoader::source This property holds the location of the source file containing the 3D asset. Changing this property will unload the current asset and attempt to load an asset from the given URL. The success or failure of the load operation is indicated by \l status. */ /*! \qmlproperty enumeration RuntimeLoader::status This property holds the status of the latest load operation. \value RuntimeLoader.Empty No URL was specified. \value RuntimeLoader.Success The load operation was successful. \value RuntimeLoader.Error The load operation failed. A human-readable error message is provided by \l errorString. \readonly */ /*! \qmlproperty string RuntimeLoader::errorString This property holds a human-readable string indicating the status of the latest load operation. \readonly */ /*! \qmlproperty Bounds RuntimeLoader::bounds This property describes the extents of the bounding volume around the imported model. \note The value may not be available before the first render \readonly */ /*! \qmlproperty Instancing RuntimeLoader::instancing If this property is set, the imported model will not be rendered normally. Instead, a number of instances will be rendered, as defined by the instance table. See the \l{Instanced Rendering} overview documentation for more information. */ QT_BEGIN_NAMESPACE QQuick3DRuntimeLoader::QQuick3DRuntimeLoader(QQuick3DNode *parent) : QQuick3DNode(parent) { } QUrl QQuick3DRuntimeLoader::source() const { return m_source; } void QQuick3DRuntimeLoader::setSource(const QUrl &newSource) { if (m_source == newSource) return; const QQmlContext *context = qmlContext(this); auto resolvedUrl = (context ? context->resolvedUrl(newSource) : newSource); if (m_source == resolvedUrl) return; m_source = resolvedUrl; emit sourceChanged(); if (isComponentComplete()) loadSource(); } void QQuick3DRuntimeLoader::componentComplete() { QQuick3DNode::componentComplete(); loadSource(); } QStringList QQuick3DRuntimeLoader::supportedExtensions() { static QStringList extensions; if (!extensions.isEmpty()) return extensions; static const QStringList supportedExtensions = { QLatin1StringView("obj"), QLatin1StringView("gltf"), QLatin1StringView("glb")}; QSSGAssetImportManager importManager; const auto types = importManager.getImporterPluginInfos(); for (const auto &t : types) { for (const QString &extension : t.inputExtensions) { if (supportedExtensions.contains(extension)) extensions << extension; } } return extensions; } #if QT_CONFIG(mimetype) QList QQuick3DRuntimeLoader::supportedMimeTypes() { static QList mimeTypes; if (!mimeTypes.isEmpty()) return mimeTypes; const QStringList &extensions = supportedExtensions(); QMimeDatabase db; for (const auto &ext : extensions) { // TODO: Change to db.mimeTypesForExtension(ext), once it is implemented (QTBUG-118566) const QString fileName = QLatin1StringView("test.") + ext; mimeTypes << db.mimeTypesForFileName(fileName); } return mimeTypes; } #endif static void boxBoundsRecursive(const QQuick3DNode *baseNode, const QQuick3DNode *node, QQuick3DBounds3 &accBounds) { if (!node) return; if (auto *model = qobject_cast(node)) { auto b = model->bounds(); for (const QVector3D point : b.bounds.toQSSGBoxPoints()) { auto p = model->mapPositionToNode(const_cast(baseNode), point); if (Q_UNLIKELY(accBounds.bounds.isEmpty())) accBounds.bounds = { p, p }; else accBounds.bounds.include(p); } } for (auto *child : node->childItems()) boxBoundsRecursive(baseNode, qobject_cast(child), accBounds); } template static void applyToModels(QQuick3DObject *obj, Func &&lambda) { if (!obj) return; for (auto *child : obj->childItems()) { if (auto *model = qobject_cast(child)) lambda(model); applyToModels(child, lambda); } } void QQuick3DRuntimeLoader::loadSource() { delete m_root; m_root.clear(); QSSGBufferManager::unregisterMeshData(m_assetId); m_status = Status::Empty; m_errorString = QStringLiteral("No file selected"); if (!m_source.isValid()) { emit statusChanged(); emit errorStringChanged(); return; } QSSGAssetImportManager importManager; QSSGSceneDesc::Scene scene; QString error(QStringLiteral("Unknown error")); auto result = importManager.importFile(m_source, scene, &error); switch (result) { case QSSGAssetImportManager::ImportState::Success: m_errorString = QStringLiteral("Success!"); m_status = Status::Success; break; case QSSGAssetImportManager::ImportState::IoError: m_errorString = QStringLiteral("IO Error: ") + error; m_status = Status::Error; break; case QSSGAssetImportManager::ImportState::Unsupported: m_errorString = QStringLiteral("Unsupported: ") + error; m_status = Status::Error; break; } if (m_status == Status::Success) { // We create a dummy root node here, as it will be the parent to the first-level nodes // and resources. If we use 'this' those first-level nodes/resources won't be deleted // when a new scene is loaded. m_root = new QQuick3DNode(this); m_imported = QSSGRuntimeUtils::createScene(*m_root, scene); m_assetId = scene.id; m_boundsDirty = true; m_instancingChanged = m_instancing != nullptr; updateModels(); // Cleanup scene before deleting. scene.cleanup(); } else { m_source.clear(); emit sourceChanged(); } emit statusChanged(); emit errorStringChanged(); } void QQuick3DRuntimeLoader::updateModels() { if (m_instancingChanged) { applyToModels(m_imported, [this](QQuick3DModel *model) { model->setInstancing(m_instancing); model->setInstanceRoot(m_imported); }); m_instancingChanged = false; } } QQuick3DRuntimeLoader::Status QQuick3DRuntimeLoader::status() const { return m_status; } QString QQuick3DRuntimeLoader::errorString() const { return m_errorString; } QSSGRenderGraphObject *QQuick3DRuntimeLoader::updateSpatialNode(QSSGRenderGraphObject *node) { auto *result = QQuick3DNode::updateSpatialNode(node); if (m_boundsDirty) QMetaObject::invokeMethod(this, &QQuick3DRuntimeLoader::boundsChanged, Qt::QueuedConnection); return result; } void QQuick3DRuntimeLoader::calculateBounds() { if (!m_imported || !m_boundsDirty) return; m_bounds.bounds.setEmpty(); boxBoundsRecursive(m_imported, m_imported, m_bounds); m_boundsDirty = false; } const QQuick3DBounds3 &QQuick3DRuntimeLoader::bounds() const { if (m_boundsDirty) { auto *that = const_cast(this); that->calculateBounds(); return that->m_bounds; } return m_bounds; } QQuick3DInstancing *QQuick3DRuntimeLoader::instancing() const { return m_instancing; } void QQuick3DRuntimeLoader::setInstancing(QQuick3DInstancing *newInstancing) { if (m_instancing == newInstancing) return; QQuick3DObjectPrivate::attachWatcher(this, &QQuick3DRuntimeLoader::setInstancing, newInstancing, m_instancing); m_instancing = newInstancing; m_instancingChanged = true; updateModels(); emit instancingChanged(); } QT_END_NAMESPACE