forked from qt-creator/qt-creator
QmlDesigner: Create thumbnail for imported 3D assets
Qmlpuppet now allows creating icon from qml source code. Invoke qmlpuppet with following arguments to create icons: --rendericon <size> <icon file name> <icon qml source> E.g. --rendericon 24 ~/my_icon.png ~/my_icon.qml Two icons are created, one with size x size dimensions and one with double the dimensions and "@2x" injected into the file name. 3D asset import utilizes this icon rendering to produce item library icons for imported components. Change-Id: I92c62c80d961f5f61a0ce1c09b32bbcbf80ea56c Fixes: QDS-1052 Reviewed-by: Mahmoud Badri <mahmoud.badri@qt.io> Reviewed-by: Thomas Hartmann <thomas.hartmann@qt.io>
This commit is contained in:
92
share/qtcreator/qml/qmlpuppet/mockfiles/IconRenderer3D.qml
Normal file
92
share/qtcreator/qml/qmlpuppet/mockfiles/IconRenderer3D.qml
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2020 The Qt Company Ltd.
|
||||||
|
** Contact: https://www.qt.io/licensing/
|
||||||
|
**
|
||||||
|
** This file is part of Qt Creator.
|
||||||
|
**
|
||||||
|
** Commercial License Usage
|
||||||
|
** Licensees holding valid commercial Qt licenses may use this file in
|
||||||
|
** accordance with the commercial license agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and The Qt Company. For licensing terms
|
||||||
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||||
|
** information use the contact form at https://www.qt.io/contact-us.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3 as published by the Free Software
|
||||||
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||||
|
** included in the packaging of this file. Please review the following
|
||||||
|
** information to ensure the GNU General Public License requirements will
|
||||||
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick3D 1.15
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: viewRoot
|
||||||
|
width: 1024
|
||||||
|
height: 1024
|
||||||
|
visible: true
|
||||||
|
|
||||||
|
property alias view3D: view3D
|
||||||
|
property alias camPos: viewCamera.position
|
||||||
|
|
||||||
|
function setSceneToBox()
|
||||||
|
{
|
||||||
|
selectionBox.targetNode = view3D.importScene;
|
||||||
|
}
|
||||||
|
|
||||||
|
function fitAndHideBox() : bool
|
||||||
|
{
|
||||||
|
cameraControl.focusObject(selectionBox.model, viewCamera.eulerRotation, true);
|
||||||
|
if (cameraControl._zoomFactor < 0.1) {
|
||||||
|
view3D.importScene.scale = view3D.importScene.scale.times(10);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (cameraControl._zoomFactor > 100) {
|
||||||
|
view3D.importScene.scale = view3D.importScene.scale.times(0.1);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
selectionBox.visible = false;
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
View3D {
|
||||||
|
id: view3D
|
||||||
|
camera: viewCamera
|
||||||
|
environment: sceneEnv
|
||||||
|
|
||||||
|
SceneEnvironment {
|
||||||
|
id: sceneEnv
|
||||||
|
antialiasingMode: SceneEnvironment.MSAA
|
||||||
|
antialiasingQuality: SceneEnvironment.VeryHigh
|
||||||
|
}
|
||||||
|
|
||||||
|
PerspectiveCamera {
|
||||||
|
id: viewCamera
|
||||||
|
position: Qt.vector3d(-200, 200, 200)
|
||||||
|
eulerRotation: Qt.vector3d(-45, -45, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
DirectionalLight {
|
||||||
|
rotation: viewCamera.rotation
|
||||||
|
}
|
||||||
|
|
||||||
|
SelectionBox {
|
||||||
|
id: selectionBox
|
||||||
|
view3D: view3D
|
||||||
|
geometryName: "SB"
|
||||||
|
}
|
||||||
|
|
||||||
|
EditCameraController {
|
||||||
|
id: cameraControl
|
||||||
|
camera: view3D.camera
|
||||||
|
view3d: view3D
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -200,7 +200,7 @@ QVector4D GeneralHelper::focusObjectToCamera(QQuick3DCamera *camera, float defau
|
|||||||
|
|
||||||
camera->setPosition(lookAt + newLookVector);
|
camera->setPosition(lookAt + newLookVector);
|
||||||
|
|
||||||
float newZoomFactor = updateZoom ? qBound(.01f, float(maxExtent / 700.), 100.f) : oldZoom;
|
float newZoomFactor = updateZoom ? qBound(.01f, float(maxExtent / 900.), 100.f) : oldZoom;
|
||||||
float cameraZoomFactor = zoomCamera(camera, 0, defaultLookAtDistance, lookAt, newZoomFactor, false);
|
float cameraZoomFactor = zoomCamera(camera, 0, defaultLookAtDistance, lookAt, newZoomFactor, false);
|
||||||
|
|
||||||
return QVector4D(lookAt, cameraZoomFactor);
|
return QVector4D(lookAt, cameraZoomFactor);
|
||||||
|
@@ -0,0 +1,181 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2020 The Qt Company Ltd.
|
||||||
|
** Contact: https://www.qt.io/licensing/
|
||||||
|
**
|
||||||
|
** This file is part of Qt Creator.
|
||||||
|
**
|
||||||
|
** Commercial License Usage
|
||||||
|
** Licensees holding valid commercial Qt licenses may use this file in
|
||||||
|
** accordance with the commercial license agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and The Qt Company. For licensing terms
|
||||||
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||||
|
** information use the contact form at https://www.qt.io/contact-us.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3 as published by the Free Software
|
||||||
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||||
|
** included in the packaging of this file. Please review the following
|
||||||
|
** information to ensure the GNU General Public License requirements will
|
||||||
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include "iconrenderer.h"
|
||||||
|
#include "../editor3d/selectionboxgeometry.h"
|
||||||
|
#include "../editor3d/generalhelper.h"
|
||||||
|
|
||||||
|
#include <QtQml/qqmlcomponent.h>
|
||||||
|
#include <QtQml/qqmlengine.h>
|
||||||
|
#include <QtQml/qqmlproperty.h>
|
||||||
|
#include <QtQml/qqmlcontext.h>
|
||||||
|
#include <QtQuick/qquickview.h>
|
||||||
|
#include <QtQuick/qquickitem.h>
|
||||||
|
#include <QtGui/qsurfaceformat.h>
|
||||||
|
#include <QtGui/qimage.h>
|
||||||
|
#include <QtGui/qguiapplication.h>
|
||||||
|
#include <QtCore/qtimer.h>
|
||||||
|
#include <QtCore/qfileinfo.h>
|
||||||
|
#include <QtCore/qdir.h>
|
||||||
|
|
||||||
|
#ifdef QUICK3D_MODULE
|
||||||
|
#include <QtQuick3D/private/qquick3dnode_p.h>
|
||||||
|
#include <QtQuick3D/private/qquick3dviewport_p.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <private/qquickdesignersupportitems_p.h>
|
||||||
|
|
||||||
|
IconRenderer::IconRenderer(int size, const QString &filePath, const QString &source)
|
||||||
|
: QObject(nullptr)
|
||||||
|
, m_size(size)
|
||||||
|
, m_filePath(filePath)
|
||||||
|
, m_source(source)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void IconRenderer::setupRender()
|
||||||
|
{
|
||||||
|
DesignerSupport::activateDesignerMode();
|
||||||
|
DesignerSupport::activateDesignerWindowManager();
|
||||||
|
|
||||||
|
m_quickView = new QQuickView;
|
||||||
|
|
||||||
|
QSurfaceFormat surfaceFormat = m_quickView->requestedFormat();
|
||||||
|
surfaceFormat.setVersion(4, 1);
|
||||||
|
surfaceFormat.setProfile(QSurfaceFormat::CoreProfile);
|
||||||
|
m_quickView->setFormat(surfaceFormat);
|
||||||
|
|
||||||
|
DesignerSupport::createOpenGLContext(m_quickView);
|
||||||
|
|
||||||
|
QQmlComponent component(m_quickView->engine());
|
||||||
|
component.loadUrl(QUrl::fromLocalFile(m_source));
|
||||||
|
QObject *iconItem = component.create();
|
||||||
|
|
||||||
|
if (iconItem) {
|
||||||
|
QQuickItem *containerItem = nullptr;
|
||||||
|
bool is3D = false;
|
||||||
|
#ifdef QUICK3D_MODULE
|
||||||
|
if (auto scene = qobject_cast<QQuick3DNode *>(iconItem)) {
|
||||||
|
qmlRegisterType<QmlDesigner::Internal::SelectionBoxGeometry>("SelectionBoxGeometry", 1, 0, "SelectionBoxGeometry");
|
||||||
|
QQmlComponent component(m_quickView->engine());
|
||||||
|
component.loadUrl(QUrl("qrc:/qtquickplugin/mockfiles/IconRenderer3D.qml"));
|
||||||
|
containerItem = qobject_cast<QQuickItem *>(component.create());
|
||||||
|
DesignerSupport::setRootItem(m_quickView, containerItem);
|
||||||
|
|
||||||
|
auto helper = new QmlDesigner::Internal::GeneralHelper();
|
||||||
|
m_quickView->engine()->rootContext()->setContextProperty("_generalHelper", helper);
|
||||||
|
|
||||||
|
m_contentItem = QQmlProperty::read(containerItem, "view3D").value<QQuickItem *>();
|
||||||
|
auto view3D = qobject_cast<QQuick3DViewport *>(m_contentItem);
|
||||||
|
view3D->setImportScene(scene);
|
||||||
|
is3D = true;
|
||||||
|
} else
|
||||||
|
#endif
|
||||||
|
if (auto scene = qobject_cast<QQuickItem *>(iconItem)) {
|
||||||
|
m_contentItem = scene;
|
||||||
|
containerItem = new QQuickItem();
|
||||||
|
containerItem->setSize(QSizeF(1024, 1024));
|
||||||
|
DesignerSupport::setRootItem(m_quickView, containerItem);
|
||||||
|
m_contentItem->setParentItem(containerItem);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (containerItem && m_contentItem) {
|
||||||
|
m_contentItem->setSize(QSizeF(m_size, m_size));
|
||||||
|
if (m_contentItem->width() > containerItem->width())
|
||||||
|
containerItem->setWidth(m_contentItem->width());
|
||||||
|
if (m_contentItem->height() > containerItem->height())
|
||||||
|
containerItem->setHeight(m_contentItem->height());
|
||||||
|
|
||||||
|
QTimer::singleShot(0, this, [this, containerItem, is3D]() {
|
||||||
|
m_designerSupport.refFromEffectItem(m_quickView->rootObject(), false);
|
||||||
|
QQuickDesignerSupportItems::disableNativeTextRendering(m_quickView->rootObject());
|
||||||
|
|
||||||
|
#ifdef QUICK3D_MODULE
|
||||||
|
if (is3D) {
|
||||||
|
// Render once to make sure scene is up to date before we set up the selection box
|
||||||
|
render({});
|
||||||
|
QMetaObject::invokeMethod(containerItem, "setSceneToBox");
|
||||||
|
bool success = false;
|
||||||
|
int tries = 0;
|
||||||
|
while (!success && tries < 10) {
|
||||||
|
++tries;
|
||||||
|
render({});
|
||||||
|
QMetaObject::invokeMethod(containerItem, "fitAndHideBox",
|
||||||
|
Q_RETURN_ARG(bool, success));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
Q_UNUSED(is3D)
|
||||||
|
#endif
|
||||||
|
QFileInfo fi(m_filePath);
|
||||||
|
|
||||||
|
// Render regular size image
|
||||||
|
render(fi.absoluteFilePath());
|
||||||
|
|
||||||
|
// Render @2x image
|
||||||
|
m_contentItem->setSize(QSizeF(m_size * 2, m_size * 2));
|
||||||
|
|
||||||
|
QString saveFile;
|
||||||
|
saveFile = fi.absolutePath() + '/' + fi.completeBaseName() + "@2x";
|
||||||
|
if (!fi.suffix().isEmpty())
|
||||||
|
saveFile += '.' + fi.suffix();
|
||||||
|
|
||||||
|
fi.absoluteDir().mkpath(".");
|
||||||
|
|
||||||
|
render(saveFile);
|
||||||
|
|
||||||
|
// Allow little time for file operations to finish
|
||||||
|
QTimer::singleShot(1000, qGuiApp, &QGuiApplication::quit);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
qGuiApp->quit();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
qGuiApp->quit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void IconRenderer::render(const QString &fileName)
|
||||||
|
{
|
||||||
|
std::function<void (QQuickItem *)> updateNodesRecursive;
|
||||||
|
updateNodesRecursive = [&updateNodesRecursive](QQuickItem *item) {
|
||||||
|
const auto childItems = item->childItems();
|
||||||
|
for (QQuickItem *childItem : childItems)
|
||||||
|
updateNodesRecursive(childItem);
|
||||||
|
DesignerSupport::updateDirtyNode(item);
|
||||||
|
};
|
||||||
|
updateNodesRecursive(m_quickView->rootObject());
|
||||||
|
|
||||||
|
QRect rect(QPoint(), m_contentItem->size().toSize());
|
||||||
|
QImage renderImage = m_designerSupport.renderImageForItem(m_quickView->rootObject(),
|
||||||
|
rect, rect.size());
|
||||||
|
if (!fileName.isEmpty()) {
|
||||||
|
QFileInfo fi(fileName);
|
||||||
|
if (fi.suffix().isEmpty())
|
||||||
|
renderImage.save(fileName, "PNG");
|
||||||
|
else
|
||||||
|
renderImage.save(fileName);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,56 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2020 The Qt Company Ltd.
|
||||||
|
** Contact: https://www.qt.io/licensing/
|
||||||
|
**
|
||||||
|
** This file is part of Qt Creator.
|
||||||
|
**
|
||||||
|
** Commercial License Usage
|
||||||
|
** Licensees holding valid commercial Qt licenses may use this file in
|
||||||
|
** accordance with the commercial license agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and The Qt Company. For licensing terms
|
||||||
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||||
|
** information use the contact form at https://www.qt.io/contact-us.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3 as published by the Free Software
|
||||||
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||||
|
** included in the packaging of this file. Please review the following
|
||||||
|
** information to ensure the GNU General Public License requirements will
|
||||||
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QtCore/qobject.h>
|
||||||
|
#include <QtCore/qstring.h>
|
||||||
|
|
||||||
|
#include <designersupportdelegate.h>
|
||||||
|
|
||||||
|
QT_BEGIN_NAMESPACE
|
||||||
|
class QQuickView;
|
||||||
|
class QQuickItem;
|
||||||
|
QT_END_NAMESPACE
|
||||||
|
|
||||||
|
class IconRenderer : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit IconRenderer(int size, const QString &filePath, const QString &source);
|
||||||
|
|
||||||
|
void setupRender();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void render(const QString &fileName);
|
||||||
|
|
||||||
|
int m_size = 16;
|
||||||
|
QString m_filePath;
|
||||||
|
QString m_source;
|
||||||
|
QQuickView *m_quickView = nullptr;
|
||||||
|
QQuickItem *m_contentItem = nullptr;
|
||||||
|
DesignerSupport m_designerSupport;
|
||||||
|
};
|
@@ -0,0 +1,3 @@
|
|||||||
|
HEADERS += $$PWD/iconrenderer.h
|
||||||
|
|
||||||
|
SOURCES += $$PWD/iconrenderer.cpp
|
@@ -13,6 +13,7 @@ include (../container/container.pri)
|
|||||||
include (../interfaces/interfaces.pri)
|
include (../interfaces/interfaces.pri)
|
||||||
include (../types/types.pri)
|
include (../types/types.pri)
|
||||||
include (../qmlprivategate/qmlprivategate.pri)
|
include (../qmlprivategate/qmlprivategate.pri)
|
||||||
|
include (iconrenderer/iconrenderer.pri)
|
||||||
|
|
||||||
SOURCES += $$PWD/qml2puppetmain.cpp
|
SOURCES += $$PWD/qml2puppetmain.cpp
|
||||||
RESOURCES += $$PWD/../qmlpuppet.qrc
|
RESOURCES += $$PWD/../qmlpuppet.qrc
|
||||||
|
@@ -32,6 +32,7 @@
|
|||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
#include <qt5nodeinstanceclientproxy.h>
|
#include <qt5nodeinstanceclientproxy.h>
|
||||||
|
#include "iconrenderer/iconrenderer.h"
|
||||||
|
|
||||||
#include <QQmlComponent>
|
#include <QQmlComponent>
|
||||||
#include <QQmlEngine>
|
#include <QQmlEngine>
|
||||||
@@ -52,11 +53,13 @@ int internalMain(QGuiApplication *application)
|
|||||||
QCoreApplication::setApplicationVersion("1.0.0");
|
QCoreApplication::setApplicationVersion("1.0.0");
|
||||||
|
|
||||||
if (application->arguments().count() < 2
|
if (application->arguments().count() < 2
|
||||||
|| (application->arguments().at(1) == "--readcapturedstream" && application->arguments().count() < 3)) {
|
|| (application->arguments().at(1) == "--readcapturedstream" && application->arguments().count() < 3)
|
||||||
|
|| (application->arguments().at(1) == "--rendericon" && application->arguments().count() < 5)) {
|
||||||
qDebug() << "Usage:\n";
|
qDebug() << "Usage:\n";
|
||||||
qDebug() << "--test";
|
qDebug() << "--test";
|
||||||
qDebug() << "--version";
|
qDebug() << "--version";
|
||||||
qDebug() << "--readcapturedstream <stream file> [control stream file]";
|
qDebug() << "--readcapturedstream <stream file> [control stream file]";
|
||||||
|
qDebug() << "--rendericon <icon size> <icon file name> <icon source qml>";
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@@ -108,7 +111,16 @@ int internalMain(QGuiApplication *application)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (application->arguments().at(1) == "--rendericon") {
|
||||||
|
int size = application->arguments().at(2).toInt();
|
||||||
|
QString iconFileName = application->arguments().at(3);
|
||||||
|
QString iconSource = application->arguments().at(4);
|
||||||
|
|
||||||
|
IconRenderer *iconRenderer = new IconRenderer(size, iconFileName, iconSource);
|
||||||
|
iconRenderer->setupRender();
|
||||||
|
|
||||||
|
return application->exec();
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_QT_BREAKPAD
|
#ifdef ENABLE_QT_BREAKPAD
|
||||||
const QString libexecPath = QCoreApplication::applicationDirPath() + '/' + RELATIVE_LIBEXEC_PATH;
|
const QString libexecPath = QCoreApplication::applicationDirPath() + '/' + RELATIVE_LIBEXEC_PATH;
|
||||||
|
@@ -7,6 +7,7 @@
|
|||||||
<file>mockfiles/SwipeView.qml</file>
|
<file>mockfiles/SwipeView.qml</file>
|
||||||
<file>mockfiles/GenericBackend.qml</file>
|
<file>mockfiles/GenericBackend.qml</file>
|
||||||
<file>mockfiles/Dialog.qml</file>
|
<file>mockfiles/Dialog.qml</file>
|
||||||
|
<file>mockfiles/IconRenderer3D.qml</file>
|
||||||
<file>mockfiles/EditView3D.qml</file>
|
<file>mockfiles/EditView3D.qml</file>
|
||||||
<file>mockfiles/EditCameraController.qml</file>
|
<file>mockfiles/EditCameraController.qml</file>
|
||||||
<file>mockfiles/Arrow.qml</file>
|
<file>mockfiles/Arrow.qml</file>
|
||||||
|
@@ -28,6 +28,7 @@
|
|||||||
|
|
||||||
#include "rewriterview.h"
|
#include "rewriterview.h"
|
||||||
#include "model.h"
|
#include "model.h"
|
||||||
|
#include "puppetcreator.h"
|
||||||
|
|
||||||
#include <QtCore/qdir.h>
|
#include <QtCore/qdir.h>
|
||||||
#include <QtCore/qdiriterator.h>
|
#include <QtCore/qdiriterator.h>
|
||||||
@@ -84,44 +85,14 @@ void ItemLibraryAssetImporter::importQuick3D(const QStringList &inputFiles,
|
|||||||
parseFiles(inputFiles, options, extToImportOptionsMap);
|
parseFiles(inputFiles, options, extToImportOptionsMap);
|
||||||
|
|
||||||
if (!isCancelled()) {
|
if (!isCancelled()) {
|
||||||
// Don't allow cancel anymore as existing asset overwrites are not trivially recoverable.
|
// Wait for icon generation processes to finish
|
||||||
// Also, on Windows at least you can't delete a subdirectory of a watched directory,
|
if (m_qmlPuppetProcesses.isEmpty()) {
|
||||||
// so complete rollback is no longer possible in any case.
|
finalizeQuick3DImport();
|
||||||
emit importNearlyFinished();
|
} else {
|
||||||
|
m_qmlPuppetCount = m_qmlPuppetProcesses.size();
|
||||||
copyImportedFiles();
|
const QString progressTitle = tr("Generating icons.");
|
||||||
|
|
||||||
auto doc = QmlDesignerPlugin::instance()->currentDesignDocument();
|
|
||||||
Model *model = doc ? doc->currentModel() : nullptr;
|
|
||||||
if (model && !m_importFiles.isEmpty()) {
|
|
||||||
const QString progressTitle = tr("Updating data model.");
|
|
||||||
addInfo(progressTitle);
|
addInfo(progressTitle);
|
||||||
notifyProgress(0, progressTitle);
|
notifyProgress(0, progressTitle);
|
||||||
|
|
||||||
// Trigger underlying qmljs snapshot update by making a non-change to the doc
|
|
||||||
model->rewriterView()->textModifier()->replace(0, 0, {});
|
|
||||||
|
|
||||||
// There is a inbuilt delay before rewriter change actually updates the data model,
|
|
||||||
// so we need to wait for a moment to allow the change to take effect.
|
|
||||||
// Otherwise subsequent subcomponent manager update won't detect new imports properly.
|
|
||||||
QTimer *timer = new QTimer(parent());
|
|
||||||
static int counter;
|
|
||||||
counter = 0;
|
|
||||||
timer->callOnTimeout([this, timer, progressTitle, doc]() {
|
|
||||||
if (!isCancelled()) {
|
|
||||||
notifyProgress(++counter * 10, progressTitle);
|
|
||||||
if (counter >= 10) {
|
|
||||||
doc->updateSubcomponentManager();
|
|
||||||
timer->stop();
|
|
||||||
notifyFinished();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
timer->stop();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
timer->start(100);
|
|
||||||
} else {
|
|
||||||
notifyFinished();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
@@ -208,6 +179,26 @@ QHash<QString, QStringList> ItemLibraryAssetImporter::supportedExtensions() cons
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ItemLibraryAssetImporter::processFinished(int exitCode, QProcess::ExitStatus exitStatus)
|
||||||
|
{
|
||||||
|
Q_UNUSED(exitCode)
|
||||||
|
Q_UNUSED(exitStatus)
|
||||||
|
|
||||||
|
auto process = qobject_cast<QProcess *>(sender());
|
||||||
|
if (process) {
|
||||||
|
m_qmlPuppetProcesses.remove(process);
|
||||||
|
process->deleteLater();
|
||||||
|
const QString progressTitle = tr("Generating icons.");
|
||||||
|
if (m_qmlPuppetProcesses.isEmpty()) {
|
||||||
|
notifyProgress(100, progressTitle);
|
||||||
|
finalizeQuick3DImport();
|
||||||
|
} else {
|
||||||
|
notifyProgress(int(100. * (1. - double(m_qmlPuppetCount) / double(m_qmlPuppetProcesses.size()))),
|
||||||
|
progressTitle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void ItemLibraryAssetImporter::notifyFinished()
|
void ItemLibraryAssetImporter::notifyFinished()
|
||||||
{
|
{
|
||||||
m_isImporting = false;
|
m_isImporting = false;
|
||||||
@@ -224,6 +215,9 @@ void ItemLibraryAssetImporter::reset()
|
|||||||
m_tempDir = new QTemporaryDir;
|
m_tempDir = new QTemporaryDir;
|
||||||
m_importFiles.clear();
|
m_importFiles.clear();
|
||||||
m_overwrittenImports.clear();
|
m_overwrittenImports.clear();
|
||||||
|
qDeleteAll(m_qmlPuppetProcesses);
|
||||||
|
m_qmlPuppetProcesses.clear();
|
||||||
|
m_qmlPuppetCount = 0;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -307,6 +301,14 @@ void ItemLibraryAssetImporter::parseQuick3DAsset(const QString &file, const QVar
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QHash<QString, QString> assetFiles;
|
||||||
|
const int outDirPathSize = outDir.path().size();
|
||||||
|
auto insertAsset = [&](const QString &filePath) {
|
||||||
|
QString targetPath = filePath.mid(outDirPathSize);
|
||||||
|
targetPath.prepend(targetDirPath);
|
||||||
|
assetFiles.insert(filePath, targetPath);
|
||||||
|
};
|
||||||
|
|
||||||
// Generate qmldir file if importer doesn't already make one
|
// Generate qmldir file if importer doesn't already make one
|
||||||
QString qmldirFileName = outDir.absoluteFilePath(QStringLiteral("qmldir"));
|
QString qmldirFileName = outDir.absoluteFilePath(QStringLiteral("qmldir"));
|
||||||
if (!QFileInfo(qmldirFileName).exists()) {
|
if (!QFileInfo(qmldirFileName).exists()) {
|
||||||
@@ -349,8 +351,6 @@ void ItemLibraryAssetImporter::parseQuick3DAsset(const QString &file, const QVar
|
|||||||
int nlIdx = content.lastIndexOf('\n', braceIdx);
|
int nlIdx = content.lastIndexOf('\n', braceIdx);
|
||||||
QByteArray rootItem = content.mid(nlIdx, braceIdx - nlIdx).trimmed();
|
QByteArray rootItem = content.mid(nlIdx, braceIdx - nlIdx).trimmed();
|
||||||
if (rootItem == "Node") { // a 3D object
|
if (rootItem == "Node") { // a 3D object
|
||||||
QFile::copy(":/ItemLibrary/images/item-3D_model-icon.png", iconFileName);
|
|
||||||
QFile::copy(":/ItemLibrary/images/item-3D_model-icon@2x.png", iconFileName2x);
|
|
||||||
// create hints file with proper hints
|
// create hints file with proper hints
|
||||||
QFile file(outDir.path() + '/' + fi.baseName() + ".hints");
|
QFile file(outDir.path() + '/' + fi.baseName() + ".hints");
|
||||||
file.open(QIODevice::WriteOnly | QIODevice::Text);
|
file.open(QIODevice::WriteOnly | QIODevice::Text);
|
||||||
@@ -359,9 +359,14 @@ void ItemLibraryAssetImporter::parseQuick3DAsset(const QString &file, const QVar
|
|||||||
out << "canBeDroppedInFormEditor: false" << endl;
|
out << "canBeDroppedInFormEditor: false" << endl;
|
||||||
out << "canBeDroppedInView3D: true" << endl;
|
out << "canBeDroppedInView3D: true" << endl;
|
||||||
file.close();
|
file.close();
|
||||||
} else {
|
}
|
||||||
QFile::copy(":/ItemLibrary/images/item-default-icon.png", iconFileName);
|
QString outIconSource = QString::fromUtf8(content);
|
||||||
QFile::copy(":/ItemLibrary/images/item-default-icon@2x.png", iconFileName2x);
|
if (generateComponentIcon(24, iconFileName, qmlIt.filePath())) {
|
||||||
|
// Since icon is generated by external process, the file won't be
|
||||||
|
// ready for asset gathering below, so assume its generation succeeds
|
||||||
|
// and add it now.
|
||||||
|
insertAsset(iconFileName);
|
||||||
|
insertAsset(iconFileName2x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -375,15 +380,10 @@ void ItemLibraryAssetImporter::parseQuick3DAsset(const QString &file, const QVar
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Gather all generated files
|
// Gather all generated files
|
||||||
const int outDirPathSize = outDir.path().size();
|
|
||||||
QDirIterator dirIt(outDir.path(), QDir::Files, QDirIterator::Subdirectories);
|
QDirIterator dirIt(outDir.path(), QDir::Files, QDirIterator::Subdirectories);
|
||||||
QHash<QString, QString> assetFiles;
|
|
||||||
while (dirIt.hasNext()) {
|
while (dirIt.hasNext()) {
|
||||||
dirIt.next();
|
dirIt.next();
|
||||||
const QString filePath = dirIt.filePath();
|
insertAsset(dirIt.filePath());
|
||||||
QString targetPath = filePath.mid(outDirPathSize);
|
|
||||||
targetPath.prepend(targetDirPath);
|
|
||||||
assetFiles.insert(filePath, targetPath);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy the original asset into a subdirectory
|
// Copy the original asset into a subdirectory
|
||||||
@@ -428,10 +428,12 @@ void ItemLibraryAssetImporter::copyImportedFiles()
|
|||||||
// by filesystem watchers.
|
// by filesystem watchers.
|
||||||
QHash<QString, QString>::const_iterator it = assetFiles.begin();
|
QHash<QString, QString>::const_iterator it = assetFiles.begin();
|
||||||
while (it != assetFiles.end()) {
|
while (it != assetFiles.end()) {
|
||||||
QDir targetDir = QFileInfo(it.value()).dir();
|
if (QFileInfo(it.key()).exists()) {
|
||||||
if (!targetDir.exists())
|
QDir targetDir = QFileInfo(it.value()).dir();
|
||||||
targetDir.mkpath(QStringLiteral("."));
|
if (!targetDir.exists())
|
||||||
QFile::copy(it.key(), it.value());
|
targetDir.mkpath(".");
|
||||||
|
QFile::copy(it.key(), it.value());
|
||||||
|
}
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
notifyProgress((100 * ++counter) / m_importFiles.size(), progressTitle);
|
notifyProgress((100 * ++counter) / m_importFiles.size(), progressTitle);
|
||||||
@@ -461,6 +463,77 @@ bool ItemLibraryAssetImporter::confirmAssetOverwrite(const QString &assetName)
|
|||||||
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes;
|
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ItemLibraryAssetImporter::generateComponentIcon(int size, const QString &iconFile,
|
||||||
|
const QString &iconSource)
|
||||||
|
{
|
||||||
|
auto doc = QmlDesignerPlugin::instance()->currentDesignDocument();
|
||||||
|
Model *model = doc ? doc->currentModel() : nullptr;
|
||||||
|
|
||||||
|
if (model) {
|
||||||
|
PuppetCreator puppetCreator(doc->currentTarget(), model);
|
||||||
|
puppetCreator.createQml2PuppetExecutableIfMissing();
|
||||||
|
QStringList puppetArgs;
|
||||||
|
puppetArgs << "--rendericon" << QString::number(size) << iconFile << iconSource;
|
||||||
|
QProcess *process = puppetCreator.createPuppetProcess(
|
||||||
|
"custom", {}, this, "", SLOT(processFinished(int, QProcess::ExitStatus)), puppetArgs);
|
||||||
|
|
||||||
|
if (process->waitForStarted(5000)) {
|
||||||
|
connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),
|
||||||
|
process, &QProcess::deleteLater);
|
||||||
|
m_qmlPuppetProcesses << process;
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
delete process;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ItemLibraryAssetImporter::finalizeQuick3DImport()
|
||||||
|
{
|
||||||
|
if (!isCancelled()) {
|
||||||
|
// Don't allow cancel anymore as existing asset overwrites are not trivially recoverable.
|
||||||
|
// Also, on Windows at least you can't delete a subdirectory of a watched directory,
|
||||||
|
// so complete rollback is no longer possible in any case.
|
||||||
|
emit importNearlyFinished();
|
||||||
|
|
||||||
|
copyImportedFiles();
|
||||||
|
|
||||||
|
auto doc = QmlDesignerPlugin::instance()->currentDesignDocument();
|
||||||
|
Model *model = doc ? doc->currentModel() : nullptr;
|
||||||
|
if (model && !m_importFiles.isEmpty()) {
|
||||||
|
const QString progressTitle = tr("Updating data model.");
|
||||||
|
addInfo(progressTitle);
|
||||||
|
notifyProgress(0, progressTitle);
|
||||||
|
|
||||||
|
// Trigger underlying qmljs snapshot update by making a non-change to the doc
|
||||||
|
model->rewriterView()->textModifier()->replace(0, 0, {});
|
||||||
|
|
||||||
|
// There is an inbuilt delay before rewriter change actually updates the data model,
|
||||||
|
// so we need to wait for a moment to allow the change to take effect.
|
||||||
|
// Otherwise subsequent subcomponent manager update won't detect new imports properly.
|
||||||
|
QTimer *timer = new QTimer(parent());
|
||||||
|
static int counter;
|
||||||
|
counter = 0;
|
||||||
|
timer->callOnTimeout([this, timer, progressTitle, doc]() {
|
||||||
|
if (!isCancelled()) {
|
||||||
|
notifyProgress(++counter * 10, progressTitle);
|
||||||
|
if (counter >= 10) {
|
||||||
|
doc->updateSubcomponentManager();
|
||||||
|
timer->stop();
|
||||||
|
notifyFinished();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
timer->stop();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
timer->start(100);
|
||||||
|
} else {
|
||||||
|
notifyFinished();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool ItemLibraryAssetImporter::isCancelled() const
|
bool ItemLibraryAssetImporter::isCancelled() const
|
||||||
{
|
{
|
||||||
keepUiAlive();
|
keepUiAlive();
|
||||||
|
@@ -29,6 +29,7 @@
|
|||||||
#include <QtCore/qstringlist.h>
|
#include <QtCore/qstringlist.h>
|
||||||
#include <QtCore/qhash.h>
|
#include <QtCore/qhash.h>
|
||||||
#include <QtCore/qjsonobject.h>
|
#include <QtCore/qjsonobject.h>
|
||||||
|
#include <QtCore/qprocess.h>
|
||||||
|
|
||||||
#include "import.h"
|
#include "import.h"
|
||||||
|
|
||||||
@@ -72,6 +73,9 @@ signals:
|
|||||||
void importNearlyFinished() const;
|
void importNearlyFinished() const;
|
||||||
void importFinished();
|
void importFinished();
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void processFinished(int exitCode, QProcess::ExitStatus exitStatus);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void notifyFinished();
|
void notifyFinished();
|
||||||
void reset();
|
void reset();
|
||||||
@@ -83,6 +87,8 @@ private:
|
|||||||
void notifyProgress(int value, const QString &text) const;
|
void notifyProgress(int value, const QString &text) const;
|
||||||
void keepUiAlive() const;
|
void keepUiAlive() const;
|
||||||
bool confirmAssetOverwrite(const QString &assetName);
|
bool confirmAssetOverwrite(const QString &assetName);
|
||||||
|
bool generateComponentIcon(int size, const QString &iconFile, const QString &iconSource);
|
||||||
|
void finalizeQuick3DImport();
|
||||||
|
|
||||||
#ifdef IMPORT_QUICK3D_ASSETS
|
#ifdef IMPORT_QUICK3D_ASSETS
|
||||||
QScopedPointer<QSSGAssetImportManager> m_quick3DAssetImporter;
|
QScopedPointer<QSSGAssetImportManager> m_quick3DAssetImporter;
|
||||||
@@ -93,5 +99,7 @@ private:
|
|||||||
bool m_cancelled = false;
|
bool m_cancelled = false;
|
||||||
QString m_importPath;
|
QString m_importPath;
|
||||||
QTemporaryDir *m_tempDir = nullptr;
|
QTemporaryDir *m_tempDir = nullptr;
|
||||||
|
QSet<QProcess *> m_qmlPuppetProcesses;
|
||||||
|
int m_qmlPuppetCount = 0;
|
||||||
};
|
};
|
||||||
} // QmlDesigner
|
} // QmlDesigner
|
||||||
|
@@ -177,7 +177,8 @@ QProcess *PuppetCreator::createPuppetProcess(const QString &puppetMode,
|
|||||||
const QString &socketToken,
|
const QString &socketToken,
|
||||||
QObject *handlerObject,
|
QObject *handlerObject,
|
||||||
const char *outputSlot,
|
const char *outputSlot,
|
||||||
const char *finishSlot) const
|
const char *finishSlot,
|
||||||
|
const QStringList &customOptions) const
|
||||||
{
|
{
|
||||||
return puppetProcess(qml2PuppetPath(m_availablePuppetType),
|
return puppetProcess(qml2PuppetPath(m_availablePuppetType),
|
||||||
qmlPuppetDirectory(m_availablePuppetType),
|
qmlPuppetDirectory(m_availablePuppetType),
|
||||||
@@ -185,7 +186,8 @@ QProcess *PuppetCreator::createPuppetProcess(const QString &puppetMode,
|
|||||||
socketToken,
|
socketToken,
|
||||||
handlerObject,
|
handlerObject,
|
||||||
outputSlot,
|
outputSlot,
|
||||||
finishSlot);
|
finishSlot,
|
||||||
|
customOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -195,7 +197,8 @@ QProcess *PuppetCreator::puppetProcess(const QString &puppetPath,
|
|||||||
const QString &socketToken,
|
const QString &socketToken,
|
||||||
QObject *handlerObject,
|
QObject *handlerObject,
|
||||||
const char *outputSlot,
|
const char *outputSlot,
|
||||||
const char *finishSlot) const
|
const char *finishSlot,
|
||||||
|
const QStringList &customOptions) const
|
||||||
{
|
{
|
||||||
auto puppetProcess = new QProcess;
|
auto puppetProcess = new QProcess;
|
||||||
puppetProcess->setObjectName(puppetMode);
|
puppetProcess->setObjectName(puppetMode);
|
||||||
@@ -228,7 +231,14 @@ QProcess *PuppetCreator::puppetProcess(const QString &puppetPath,
|
|||||||
if (forceFreeType)
|
if (forceFreeType)
|
||||||
forceFreeTypeOption = "-platform windows:fontengine=freetype";
|
forceFreeTypeOption = "-platform windows:fontengine=freetype";
|
||||||
|
|
||||||
puppetProcess->start(puppetPath, {socketToken, puppetMode, "-graphicssystem raster", forceFreeTypeOption });
|
if (puppetMode == "custom") {
|
||||||
|
QStringList args = customOptions;
|
||||||
|
args << "-graphicssystem raster";
|
||||||
|
args << forceFreeTypeOption;
|
||||||
|
puppetProcess->start(puppetPath, args);
|
||||||
|
} else {
|
||||||
|
puppetProcess->start(puppetPath, {socketToken, puppetMode, "-graphicssystem raster", forceFreeTypeOption });
|
||||||
|
}
|
||||||
|
|
||||||
#ifndef QMLDESIGNER_TEST
|
#ifndef QMLDESIGNER_TEST
|
||||||
QString debugPuppet = m_designerSettings.value(DesignerSettingsKey::
|
QString debugPuppet = m_designerSettings.value(DesignerSettingsKey::
|
||||||
|
@@ -57,7 +57,8 @@ public:
|
|||||||
const QString &socketToken,
|
const QString &socketToken,
|
||||||
QObject *handlerObject,
|
QObject *handlerObject,
|
||||||
const char *outputSlot,
|
const char *outputSlot,
|
||||||
const char *finishSlot) const;
|
const char *finishSlot,
|
||||||
|
const QStringList &customOptions = {}) const;
|
||||||
|
|
||||||
void setQrcMappingString(const QString qrcMapping);
|
void setQrcMappingString(const QString qrcMapping);
|
||||||
|
|
||||||
@@ -87,7 +88,8 @@ protected:
|
|||||||
const QString &socketToken,
|
const QString &socketToken,
|
||||||
QObject *handlerObject,
|
QObject *handlerObject,
|
||||||
const char *outputSlot,
|
const char *outputSlot,
|
||||||
const char *finishSlot) const;
|
const char *finishSlot,
|
||||||
|
const QStringList &customOptions) const;
|
||||||
|
|
||||||
QProcessEnvironment processEnvironment() const;
|
QProcessEnvironment processEnvironment() const;
|
||||||
|
|
||||||
|
@@ -120,6 +120,12 @@ extend_qtc_executable(qml2puppet
|
|||||||
linegeometry.cpp linegeometry.h
|
linegeometry.cpp linegeometry.h
|
||||||
)
|
)
|
||||||
|
|
||||||
|
extend_qtc_executable(qml2puppet
|
||||||
|
SOURCES_PREFIX "${SRCDIR}/qml2puppet/iconrenderer"
|
||||||
|
SOURCES
|
||||||
|
iconrenderer.cpp iconrenderer.h
|
||||||
|
)
|
||||||
|
|
||||||
extend_qtc_executable(qml2puppet
|
extend_qtc_executable(qml2puppet
|
||||||
SOURCES_PREFIX "${SRCDIR}/qml2puppet/instances"
|
SOURCES_PREFIX "${SRCDIR}/qml2puppet/instances"
|
||||||
SOURCES
|
SOURCES
|
||||||
|
@@ -217,6 +217,8 @@ QtcTool {
|
|||||||
"editor3d/selectionboxgeometry.h",
|
"editor3d/selectionboxgeometry.h",
|
||||||
"editor3d/linegeometry.cpp",
|
"editor3d/linegeometry.cpp",
|
||||||
"editor3d/linegeometry.h",
|
"editor3d/linegeometry.h",
|
||||||
|
"iconrenderer/iconrenderer.cpp",
|
||||||
|
"iconrenderer/iconrenderer.h",
|
||||||
"qml2puppetmain.cpp",
|
"qml2puppetmain.cpp",
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user