forked from qt-creator/qt-creator
Recent changes to custom geometry handling now require that the zoom process in icon creation does each focus step asynchronously, so that selection box geometry has a chance to update between frames. Fixes: QDS-4652 Change-Id: If92bf580a556a68c10d3af1406c2eabef530254a Reviewed-by: Mahmoud Badri <mahmoud.badri@qt.io>
317 lines
11 KiB
C++
317 lines
11 KiB
C++
/****************************************************************************
|
|
**
|
|
** 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
|
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
|
#include <QtGui/private/qrhi_p.h>
|
|
#include <QtQuick/private/qquickrendercontrol_p.h>
|
|
#include <QtQuick/private/qquickrendertarget_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();
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
DesignerSupport::activateDesignerWindowManager();
|
|
#endif
|
|
|
|
QQmlEngine *engine = nullptr;
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
auto view = new QQuickView;
|
|
engine = view->engine();
|
|
m_window = view;
|
|
QSurfaceFormat surfaceFormat = view->requestedFormat();
|
|
surfaceFormat.setVersion(4, 1);
|
|
surfaceFormat.setProfile(QSurfaceFormat::CoreProfile);
|
|
view->setFormat(surfaceFormat);
|
|
DesignerSupport::createOpenGLContext(view);
|
|
#else
|
|
engine = new QQmlEngine;
|
|
m_renderControl = new QQuickRenderControl;
|
|
m_window = new QQuickWindow(m_renderControl);
|
|
m_window->setDefaultAlphaBuffer(true);
|
|
m_window->setColor(Qt::transparent);
|
|
m_renderControl->initialize();
|
|
#endif
|
|
|
|
QQmlComponent component(engine);
|
|
component.loadUrl(QUrl::fromLocalFile(m_source));
|
|
QObject *iconItem = component.create();
|
|
|
|
if (iconItem) {
|
|
#ifdef QUICK3D_MODULE
|
|
if (auto scene = qobject_cast<QQuick3DNode *>(iconItem)) {
|
|
qmlRegisterType<QmlDesigner::Internal::SelectionBoxGeometry>("SelectionBoxGeometry", 1, 0, "SelectionBoxGeometry");
|
|
QQmlComponent component(engine);
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
component.loadUrl(QUrl("qrc:/qtquickplugin/mockfiles/qt5/IconRenderer3D.qml"));
|
|
m_containerItem = qobject_cast<QQuickItem *>(component.create());
|
|
DesignerSupport::setRootItem(view, m_containerItem);
|
|
#else
|
|
component.loadUrl(QUrl("qrc:/qtquickplugin/mockfiles/qt6/IconRenderer3D.qml"));
|
|
m_containerItem = qobject_cast<QQuickItem *>(component.create());
|
|
m_window->contentItem()->setSize(m_containerItem->size());
|
|
m_window->setGeometry(0, 0, m_containerItem->width(), m_containerItem->height());
|
|
m_containerItem->setParentItem(m_window->contentItem());
|
|
#endif
|
|
|
|
auto helper = new QmlDesigner::Internal::GeneralHelper();
|
|
engine->rootContext()->setContextProperty("_generalHelper", helper);
|
|
|
|
m_contentItem = QQmlProperty::read(m_containerItem, "view3D").value<QQuickItem *>();
|
|
auto view3D = qobject_cast<QQuick3DViewport *>(m_contentItem);
|
|
view3D->setImportScene(scene);
|
|
m_is3D = true;
|
|
} else
|
|
#endif
|
|
if (auto scene = qobject_cast<QQuickItem *>(iconItem)) {
|
|
m_contentItem = scene;
|
|
m_containerItem = new QQuickItem();
|
|
m_containerItem->setSize(QSizeF(1024, 1024));
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
DesignerSupport::setRootItem(view, m_containerItem);
|
|
#else
|
|
m_window->contentItem()->setSize(m_containerItem->size());
|
|
m_window->setGeometry(0, 0, m_containerItem->width(), m_containerItem->height());
|
|
m_containerItem->setParentItem(m_window->contentItem());
|
|
#endif
|
|
m_contentItem->setParentItem(m_containerItem);
|
|
}
|
|
|
|
if (m_containerItem && m_contentItem) {
|
|
resizeContent(m_size);
|
|
if (!initRhi())
|
|
QTimer::singleShot(0, qGuiApp, &QGuiApplication::quit);
|
|
QTimer::singleShot(0, this, &IconRenderer::startCreateIcon);
|
|
} else {
|
|
QTimer::singleShot(0, qGuiApp, &QGuiApplication::quit);
|
|
}
|
|
} else {
|
|
QTimer::singleShot(0, qGuiApp, &QGuiApplication::quit);
|
|
}
|
|
}
|
|
|
|
void IconRenderer::startCreateIcon()
|
|
{
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
m_designerSupport.refFromEffectItem(m_containerItem, false);
|
|
#endif
|
|
QQuickDesignerSupportItems::disableNativeTextRendering(m_containerItem);
|
|
|
|
if (m_is3D)
|
|
QTimer::singleShot(0, this, &IconRenderer::focusCamera);
|
|
else
|
|
QTimer::singleShot(0, this, &IconRenderer::finishCreateIcon);
|
|
}
|
|
|
|
void IconRenderer::focusCamera()
|
|
{
|
|
#ifdef QUICK3D_MODULE
|
|
if (m_focusStep >= 10) {
|
|
QTimer::singleShot(0, this, &IconRenderer::finishCreateIcon);
|
|
return;
|
|
}
|
|
|
|
render({});
|
|
|
|
if (m_focusStep == 0) {
|
|
QMetaObject::invokeMethod(m_containerItem, "setSceneToBox");
|
|
} else if (m_focusStep > 1 && m_focusStep < 10) {
|
|
QMetaObject::invokeMethod(m_containerItem, "fitAndHideBox");
|
|
}
|
|
++m_focusStep;
|
|
QTimer::singleShot(0, this, &IconRenderer::focusCamera);
|
|
#endif
|
|
}
|
|
|
|
void IconRenderer::finishCreateIcon()
|
|
{
|
|
QFileInfo fi(m_filePath);
|
|
|
|
// Render regular size image
|
|
render(fi.absoluteFilePath());
|
|
|
|
// Render @2x image
|
|
resizeContent(m_size * 2);
|
|
if (!initRhi())
|
|
QTimer::singleShot(1000, qGuiApp, &QGuiApplication::quit);
|
|
|
|
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);
|
|
}
|
|
|
|
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);
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
DesignerSupport::updateDirtyNode(item);
|
|
#else
|
|
if (item->flags() & QQuickItem::ItemHasContents)
|
|
item->update();
|
|
#endif
|
|
};
|
|
updateNodesRecursive(m_containerItem);
|
|
|
|
QRect rect(QPoint(), m_contentItem->size().toSize());
|
|
QImage renderImage;
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
renderImage = m_designerSupport.renderImageForItem(m_containerItem, rect, rect.size());
|
|
#else
|
|
m_renderControl->polishItems();
|
|
m_renderControl->beginFrame();
|
|
m_renderControl->sync();
|
|
m_renderControl->render();
|
|
|
|
bool readCompleted = false;
|
|
QRhiReadbackResult readResult;
|
|
readResult.completed = [&] {
|
|
readCompleted = true;
|
|
QImage wrapperImage(reinterpret_cast<const uchar *>(readResult.data.constData()),
|
|
readResult.pixelSize.width(), readResult.pixelSize.height(),
|
|
QImage::Format_RGBA8888_Premultiplied);
|
|
if (m_rhi->isYUpInFramebuffer())
|
|
renderImage = wrapperImage.mirrored().copy(0, 0, rect.width(), rect.height());
|
|
else
|
|
renderImage = wrapperImage.copy(0, 0, rect.width(), rect.height());
|
|
};
|
|
QRhiResourceUpdateBatch *readbackBatch = m_rhi->nextResourceUpdateBatch();
|
|
readbackBatch->readBackTexture(m_texture, &readResult);
|
|
|
|
QQuickRenderControlPrivate *rd = QQuickRenderControlPrivate::get(m_renderControl);
|
|
rd->cb->resourceUpdate(readbackBatch);
|
|
|
|
m_renderControl->endFrame();
|
|
#endif
|
|
if (!fileName.isEmpty()) {
|
|
QFileInfo fi(fileName);
|
|
if (fi.suffix().isEmpty())
|
|
renderImage.save(fileName, "PNG");
|
|
else
|
|
renderImage.save(fileName);
|
|
}
|
|
}
|
|
|
|
void IconRenderer::resizeContent(int dimensions)
|
|
{
|
|
QSizeF size(dimensions, dimensions);
|
|
m_contentItem->setSize(size);
|
|
if (m_contentItem->width() > m_containerItem->width())
|
|
m_containerItem->setWidth(m_contentItem->width());
|
|
if (m_contentItem->height() > m_containerItem->height())
|
|
m_containerItem->setHeight(m_contentItem->height());
|
|
}
|
|
|
|
bool IconRenderer::initRhi()
|
|
{
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
|
if (!m_rhi) {
|
|
QQuickRenderControlPrivate *rd = QQuickRenderControlPrivate::get(m_renderControl);
|
|
m_rhi = rd->rhi;
|
|
if (!m_rhi) {
|
|
qWarning() << __FUNCTION__ << "Rhi is null";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Don't bother deleting old ones as iconrender is a single shot executable
|
|
m_texTarget = nullptr;
|
|
m_rpDesc = nullptr;
|
|
m_buffer = nullptr;
|
|
m_texture = nullptr;
|
|
|
|
const QSize size = m_containerItem->size().toSize();
|
|
m_texture = m_rhi->newTexture(QRhiTexture::RGBA8, size, 1,
|
|
QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource);
|
|
if (!m_texture->create()) {
|
|
qWarning() << __FUNCTION__ << "QRhiTexture creation failed";
|
|
return false;
|
|
}
|
|
|
|
m_buffer = m_rhi->newRenderBuffer(QRhiRenderBuffer::DepthStencil, size, 1);
|
|
if (!m_buffer->create()) {
|
|
qWarning() << __FUNCTION__ << "Depth/stencil buffer creation failed";
|
|
return false;
|
|
}
|
|
|
|
QRhiTextureRenderTargetDescription rtDesc {QRhiColorAttachment(m_texture)};
|
|
rtDesc.setDepthStencilBuffer(m_buffer);
|
|
m_texTarget = m_rhi->newTextureRenderTarget(rtDesc);
|
|
m_rpDesc = m_texTarget->newCompatibleRenderPassDescriptor();
|
|
m_texTarget->setRenderPassDescriptor(m_rpDesc);
|
|
if (!m_texTarget->create()) {
|
|
qWarning() << __FUNCTION__ << "Texture render target creation failed";
|
|
return false;
|
|
}
|
|
|
|
// redirect Qt Quick rendering into our texture
|
|
m_window->setRenderTarget(QQuickRenderTarget::fromRhiRenderTarget(m_texTarget));
|
|
#endif
|
|
return true;
|
|
}
|